11632 lines
758 KiB
HTML
11632 lines
758 KiB
HTML
<html lang="en">
|
||
|
||
<head>
|
||
<meta charset="UTF-8">
|
||
<title>RanchiMall Content Collaboration Output</title>
|
||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||
<link
|
||
href="https://fonts.googleapis.com/css2?family=Playfair+Display:ital,wght@0,400;0,500;0,600;0,700;0,900;1,400;1,700;1,900&family=Poppins:wght@400;500;600;700&display=swap"
|
||
rel="stylesheet">
|
||
<link href="https://fonts.googleapis.com/css2?family=Lora:wght@400;500;600;700&display=swap" rel="stylesheet">
|
||
<style>
|
||
* {
|
||
padding: 0;
|
||
margin: 0;
|
||
box-sizing: border-box;
|
||
}
|
||
|
||
.hide {
|
||
display: none;
|
||
}
|
||
|
||
body {
|
||
font-family: 'lora', serif;
|
||
font-size: 16px;
|
||
color: #2f2f2f;
|
||
background-color: #f9f7f1;
|
||
}
|
||
|
||
header {
|
||
font-family: 'Playfair Display', serif;
|
||
font-weight: 900;
|
||
text-align: center;
|
||
font-size: 4rem;
|
||
text-transform: uppercase;
|
||
padding: 1rem 0;
|
||
letter-spacing: 6;
|
||
}
|
||
|
||
p {
|
||
margin: 1.2rem 0;
|
||
overflow-wrap: break-word;
|
||
word-wrap: break-word;
|
||
}
|
||
|
||
p:not(:first-of-type)::first-letter {
|
||
margin-left: 4rem;
|
||
}
|
||
|
||
#homepage {
|
||
display: grid;
|
||
}
|
||
|
||
.content {
|
||
display: grid;
|
||
max-width: calc(100% - 2.4rem);
|
||
}
|
||
|
||
.head {
|
||
display: flex;
|
||
flex-direction: column;
|
||
text-align: center;
|
||
justify-content: center;
|
||
position: relative;
|
||
margin-bottom: 1rem;
|
||
|
||
}
|
||
|
||
.subhead {
|
||
display: flex;
|
||
flex-direction: column;
|
||
align-items: center;
|
||
text-transform: uppercase;
|
||
padding: 0.6rem 0;
|
||
}
|
||
|
||
.subhead::after {
|
||
content: '';
|
||
position: absolute;
|
||
bottom: 0;
|
||
width: 10%;
|
||
height: 0.1rem;
|
||
background: #2f2f2f;
|
||
}
|
||
|
||
.columns {
|
||
padding-top: 2rem;
|
||
display: grid;
|
||
width: calc(100% - 3rem);
|
||
grid-template-columns: repeat(auto-fit, minmax(26rem, 1fr));
|
||
}
|
||
|
||
.column {
|
||
font-size: 1.2rem;
|
||
line-height: 1.8;
|
||
display: flex;
|
||
flex-direction: column;
|
||
padding: 0 1.5rem;
|
||
width: calc(100% - 3rem);
|
||
}
|
||
|
||
.column .headline {
|
||
text-align: center;
|
||
line-height: normal;
|
||
font-family: 'lora', serif;
|
||
display: flex;
|
||
}
|
||
|
||
.column .headline.hl1 {
|
||
font-weight: 700;
|
||
font-size: 30px;
|
||
text-transform: uppercase;
|
||
padding: 0.8rem 0;
|
||
|
||
}
|
||
|
||
.column .headline.hl2 {
|
||
font-weight: 400;
|
||
font-size: 24px;
|
||
padding: 0.8rem 0;
|
||
}
|
||
|
||
.column .headline.hl2:before {
|
||
border-top: 1px solid #2f2f2f;
|
||
content: '';
|
||
width: 100px;
|
||
height: 7px;
|
||
display: flex;
|
||
margin: 0 auto;
|
||
}
|
||
|
||
.column .headline.hl2:after {
|
||
border-bottom: 1px solid #2f2f2f;
|
||
content: '';
|
||
width: 100px;
|
||
height: 13px;
|
||
display: flex;
|
||
margin: 0 auto;
|
||
|
||
}
|
||
|
||
.column .headline.hl3 {
|
||
font-weight: 400;
|
||
font-size: 36px;
|
||
padding: 0.8rem 0;
|
||
font-style: italic;
|
||
font-family: 'Playfair Display', serif;
|
||
}
|
||
|
||
.column .headline.hl4 {
|
||
font-weight: 500;
|
||
font-size: 0.9rem;
|
||
padding: 0.8rem 0;
|
||
}
|
||
|
||
.column .headline.hl4:before {
|
||
border-top: 1px solid #2f2f2f;
|
||
content: '';
|
||
width: 100px;
|
||
height: 7px;
|
||
display: flex;
|
||
margin: 0 auto;
|
||
}
|
||
|
||
.column .headline.hl4:after {
|
||
border-bottom: 1px solid #2f2f2f;
|
||
content: '';
|
||
width: 100px;
|
||
height: 10px;
|
||
display: flex;
|
||
margin: 0 auto;
|
||
|
||
}
|
||
|
||
.topnav {
|
||
overflow: hidden;
|
||
}
|
||
|
||
/* Style the links inside the navigation bar */
|
||
.topnav a {
|
||
float: left;
|
||
color: #f2f2f2;
|
||
text-align: center;
|
||
padding: 14px 16px;
|
||
text-decoration: none;
|
||
font-size: 17px;
|
||
}
|
||
|
||
.topnav a img {
|
||
width: 12rem;
|
||
}
|
||
|
||
/*________________________________________________________________________________________________________________________________*/
|
||
/*MEDIAQUERIES*/
|
||
|
||
@media all and (max-width: 640px) {
|
||
header {
|
||
font-size: 2.5rem;
|
||
}
|
||
|
||
.topnav a img {
|
||
width: 8rem;
|
||
}
|
||
}
|
||
|
||
@media all and (min-width: 640px) {
|
||
#bodydiv {
|
||
margin: 0 12vw;
|
||
}
|
||
|
||
.columns {
|
||
margin: 0 2vw;
|
||
}
|
||
}
|
||
|
||
@media all and (min-width: 1280px) {
|
||
#bodydiv {
|
||
margin: 0 20vw;
|
||
}
|
||
|
||
.columns {
|
||
margin: 0 2vw;
|
||
}
|
||
}
|
||
|
||
@media all and (min-width: 1920px) {
|
||
#bodydiv {
|
||
margin: 0 32vw;
|
||
}
|
||
|
||
.columns {
|
||
margin: 0 4vw;
|
||
}
|
||
}
|
||
</style>
|
||
</head>
|
||
|
||
<body translate="no" onload="onLoadStartUp()">
|
||
|
||
<div class="topnav">
|
||
<a href="https://ranchimall.github.io/articles">
|
||
<img id='ranchimalltimes-logo' src=""
|
||
alt="" />
|
||
</a>
|
||
</div>
|
||
|
||
<div id="bodydiv">
|
||
<div class="head">
|
||
<div>
|
||
<header id='title'>Political History of Bengal</header>
|
||
</div>
|
||
<div class="subhead" id='subtitle'>Exported by RanchiMall Content Collaboration on FLO Blockchain</div>
|
||
</div>
|
||
<div class="content" id='sectionmaster'>
|
||
<div class="columns">
|
||
<div class="column" id='section1'>
|
||
<div class="head">
|
||
<div class="headline hl3">From Murshid to Mamata: A Political History of
|
||
Bengal</div>
|
||
</div>
|
||
<p>Murshid Quli Khan was the first Nawab of Bengal.
|
||
He was originally born a Hindu as Surya Narayan Mishra.
|
||
The city of Murshidabad was named after him.
|
||
He was Nawab of Bengal from 1717 to 1727.
|
||
Unlike other Islamic rulers, Murshid Quli Khan had only one wife, Nasiri Banu Begum, and no
|
||
concubines. He had three children, two daughters and one son</p>
|
||
<p>Murshid Quli Khan Character:
|
||
He took no delight in hunting; He never indulged in wine nor in any intoxicating drugs; neither
|
||
did
|
||
he amuse himself with singers or dancers. He always kept constant to one lawful wife, and, out
|
||
of
|
||
excess of delicacy, would not suffer any strange woman, or even eunuchs, to enter the apartments
|
||
of
|
||
his seraglio - so much so, that any slave girl once out of the harem would not be
|
||
allowed to re-enter it. He possessed very extensive learning, and paid great respect to men who
|
||
were
|
||
eminent for their piety
|
||
or erudition. He was a brave soldier, a liberal benefactor, upright and just in his dealings,
|
||
and a
|
||
steady protector of the weak.
|
||
He made no retrenchments in any royal grant, nor in those of any former Subahdar, for charitable
|
||
purposes.
|
||
He despised all the refinements of luxury, particularly in dress; no highly seasoned dishes were
|
||
served on his table.
|
||
He slept but little, and carefully observed the stated times for prayer. From breakfast till
|
||
noon he
|
||
employed himself in copying the Koran, and in administering justice; and every year he sent
|
||
Korans
|
||
of his own writing, with valuable offerings, to Mecca, Medina, Najaf, and other holy places.
|
||
|
||
|
||
FROM: https://murshidabad.net/history/history-topic-murshid-quli-khan.htm</p>
|
||
<p>Bidhan Chandra Roy was the Chief Minister of Bengal from 1948 to 1962 until his death. Dr. Roy
|
||
was
|
||
awarded with Bharat Ratna on Feb4, 1961.
|
||
|
||
He was a renowned Indian doctor, an educationist, a philanthropist and a freedom fighter. He is
|
||
considered to be the founder of five very prominent cities of Bengal; Durgapur, Howrah,
|
||
Ashoknagar,
|
||
Kalyani and Bidhannagar.
|
||
|
||
National Doctor's Day in India is celebrated each year in his memory, as he was the only few
|
||
people
|
||
from history who obtained a degree in FRCS and MRCP.
|
||
|
||
Dr. Roy was born in a bengali family in Bankipore, Patna, with his father working as an Excise
|
||
Inspector and a very pious and social worker mother.
|
||
|
||
The partition of Bengal was happening when Bidhan was still in his college, he was emotionally
|
||
touched by this but he controlled his emotions and chose to continue on his education, so he
|
||
could
|
||
serve the nation better this way.
|
||
|
||
One of the example of his strict determination is when he went to St. Bartholomev's Hospital,
|
||
England with only 1200rs and applied for his masters, he was kept being rejected by the Dean
|
||
because
|
||
he was an Asian. But Dr. Roy didn't stop by this, and after applying 30 times in a row, the Dean
|
||
had
|
||
to accept his application. After completing his Post Graduate, he became a member of Royal
|
||
College
|
||
of Physicians and a fellow of Royal College of Surgeons.
|
||
|
||
Dr. Roy then believed that Swaraj (the call for India's freedom) would remain a dream if people
|
||
were unhealthy with their body and mind. He taught and contributed at many medical institutions
|
||
like
|
||
Calcutta Medical College, Campbell Medical School, Carmichael Medical College, Jadavpur TB
|
||
Hospital,
|
||
Chittaranjan Seva Sadan, Kamla Nehru Memorial Hospital, Victoria Institution, and Chittaranjan
|
||
Cancer Hospital.
|
||
|
||
In 1942, when Rangoon fell to Japanese attack, it caused an exodus of people fearing Japanese
|
||
invasion, Dr. Roy was the Vice-Chancellor of the University of Calcutta. He there acquired
|
||
air-raid
|
||
shelters for schools and colleges to have their classes and also provided shelter and relief for
|
||
students, teachers and other employees. He was awarded Doctorate of Science in 1944 for his
|
||
noble
|
||
and dynamic contributions.
|
||
|
||
"My young friends, you are soldiers in the battle of freedom - freedom from want, fear,
|
||
ignorance,
|
||
frustration and helplessness. By a dint of hard work for the country, rendered in a spirit of
|
||
selfless service, may you march ahead with hope and courage". These was the words he spoke to
|
||
the
|
||
young crowd at a convocation in University of Lucknow in 1956.
|
||
|
||
He was also a friend and doctor of Mahatma Gandhi. Once when Gandhi was going under some fast
|
||
and
|
||
got unwell, Dr. Roy came with some medicine to treat him and Gandhi refused to take those
|
||
medicines
|
||
citing they are not Made in India and also told Dr. Roy that why don't he treats the fellow
|
||
Indian
|
||
citizens free?
|
||
In reply of this, Dr. Roy said to Gandhi "I have come here to treat someone who represents the
|
||
four
|
||
hundred million people of my country" and Gandhi accepted his medicines.
|
||
|
||
Post Independence in India, Congress Party proposed his name for Chief Minister of Bengal, which
|
||
he
|
||
refused but later accepted after Gandhiji's advice and became the Chief Minister of Bengal on 23
|
||
January, 1948. Bengal was seeing a pool of problems including communal violence, food shortage,
|
||
large flow of refugees in Bengal, unemployment etc. Dr. Roy did an amendable work by reuniting
|
||
the
|
||
party ranks, worked for every aspects he could and within 3 years normalcy settled down in
|
||
Bengal.
|
||
|
||
|
||
Just after receiving the most prestigious award of Bharat Ratna in 1961, one year later he died
|
||
on 1
|
||
July 1962 on his 80th birthday. He had gifted his house to be run as a nursing home in the name
|
||
of
|
||
his mother, Aghorkamini Devi. He had also formed a trust for his properties in Patna to carry
|
||
out
|
||
social service, with Ganga Sharan Singh, a eminent nationalist, being it's first trustee.
|
||
|
||
In memory of his excellence, The B. C. Roy National Award was introduced in 1962 and has been
|
||
awarding people for their excellent contributions in the field of medicine, politics, science,
|
||
philosophy, literature and arts, annually since 1976.</p>
|
||
<p>Jyoti Basu, (cofounder of Communist Party of India) who almost became the Prime Minister.
|
||
|
||
Jyoti Basu was the Chief Minister of Bengal from 1977 to 2000, for a record 23 years 4 months 17
|
||
days. He is known as the beacon of communism in Bengal. He would be the India's first Bengali
|
||
and
|
||
Marxist Prime Minister if CPI(M) hadn't decided to not join the government.
|
||
|
||
He was born at 43/1 Harrison Road Calcutta (Now Mahatma Gandhi Road). His father was a doctor at
|
||
Bardi village, Dhaka (Now Bangladesh).
|
||
|
||
His birth name was Jyotirindra Basu but his father changed his name in school and he became
|
||
Jyoti
|
||
Basu. He completed his bachelors in English Literature from Presidency College in Kolkata and
|
||
did
|
||
Law from England.
|
||
|
||
Having developed a strong belief in the Communist ideals, Basu returned to India in 1940 and
|
||
joined
|
||
the Communist Party of India, he also became the secretary of the Friends of the Soviet Union
|
||
and
|
||
Anti-Fascist Writers’ and Artists’ Association in Calcutta.
|
||
|
||
In 1944, Basu started working with trade and railways unions of Bengal. He established the
|
||
Bengal
|
||
Nagpur Railway Workers’ Union, and became its General Secretary. He was elected to the central
|
||
committee of the CPI Bengal in 1951.
|
||
|
||
Communist Party of India knew the loss and corruption done by congress, they formed an
|
||
anti-congress
|
||
front with Socialist Republican Paarty, Bolshevik Party of India and the Forward Bloc, no doubt
|
||
Jyoti Basu was a member of this front.
|
||
|
||
After the India-China War of 1962, differences arose in the CPI, leading Basu co-found
|
||
Communist
|
||
Party of India (Marxist).
|
||
|
||
|
||
When Basu came into power as a Chief Minister of Bengal, it was congress in ruling before him.
|
||
Basu
|
||
was a hardcore democratic supporter, he ordered probes into many matters which occurred during
|
||
the
|
||
previous congress govt. such as inordinate increase in prices of essential commodities, release
|
||
of
|
||
political prisoners, atrocities against anyone done by govt. He abolished the nefarious
|
||
Maintenance
|
||
of Internal Security Act which gave a full hand to the Congress govt to arrest anyone on mere
|
||
suspicions in the name of security.
|
||
|
||
|
||
As the Chief Minister, he from the beginning took bold reforming steps for Bengal. He
|
||
industrialized
|
||
programs for small, medium and cottage industries. Unemployment allowances, stoppage of
|
||
retrenchment
|
||
of workers, cutting down on the tax of essential commodities.
|
||
|
||
When Jyoti Basu started administrating Bengal, within 3years, nearly more than 10 lakh acre more
|
||
ceiling surplus land was identified and three-fourth of this land was distributed within a year.
|
||
This resulted in a distribution of surplus lands to landless people at such a large scale alone
|
||
in
|
||
Bengal, than that of few states of India combined to date. Bengal was the only state which put a
|
||
ceiling on landholding from the beginning which Basu changed after coming to a ruling position.
|
||
|
||
His innovative strategy was to take administration to village levels, where someone with oral
|
||
evidence could challenge the paper evidence in courts, and the powerful landlords could not
|
||
illegally acquired the lands anymore.
|
||
All measures of land reforms were taken together by Basu including distribution of homestead
|
||
lands.
|
||
|
||
|
||
It was a golden period for Bengal in 1980s when Bengal was marching ahead with a 4.2% highest
|
||
annual
|
||
rate of foodgrain growth comparing to other states at 2.5% average.
|
||
|
||
|
||
According to National Sample Survey, when CPI(M) came in rule with Basu as Chief Minister, it
|
||
was
|
||
40% of the population of whole Bengal who could not spend enough money to get 1800 calories
|
||
energy.
|
||
After 15 years to this time, Bengal saw a 17% decrease in this population. This was the biggest
|
||
reduction of extreme poverty anywhere in India over any period.
|
||
In the rural areas it dropped from 67% to 42% in fifteen years.
|
||
|
||
After the incident of Babri Mosque demolition, there was a rise of communal violence elsewhere
|
||
in
|
||
the country including the capital Delhi in 1984, but Bengal remained free of any communal
|
||
violence,
|
||
which is seen as a major achievement of Jyoti Basu's ruling period.
|
||
|
||
|
||
He had brought big initiatives in the history of Bengal, such as land reforms, minimum wages for
|
||
agricultural labourers, a three-tier panchayati system, dole for the unemployed and widows, and
|
||
the
|
||
establishment of a separate department for youth services.
|
||
|
||
Basu retired from politics in 2000, though he remained a member of the CPI(M) politburo till
|
||
2008.
|
||
</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section2'>
|
||
<div class="head">
|
||
<div class="headline hl3">Hooghly, Calcutta, 2020, Painting Hooghly with words-Build
|
||
background and bridge to Writers Building</div>
|
||
</div>
|
||
<p>For the traveler with limited time, the best way to explore Calcutta is roughly to trace the
|
||
route of
|
||
the Hooghly, meandering on and off the main thoroughfares by foot, tram and subway, known here
|
||
as
|
||
the Metro. This is not a luxury destination. It is more a journey through the grimy layers of
|
||
time.
|
||
History is inscribed on every lane, like tattoos on a body. Calcutta was once quite a diva.</p>
|
||
<p>On the left side of Calcutta the iron trusses of Howrah Bridge towers over the Hooghly river. The
|
||
pale yellow waters are dotted by Bengali canopy boats and you see occasional steel riverboats
|
||
ferrying passengers from Howrah Railway Station to Fairlie Place in Kolkata side across the
|
||
river.
|
||
An alighting passenger has to walk just a block across Fairlie Palace, and he will see a grand
|
||
red
|
||
greco roman styled three storied building that holds keys to the power in West Bengal. On the
|
||
first
|
||
floor of the building towards the old Calcutta Stock Exchange lies the chambers of the Chief
|
||
Minister.</p>
|
||
<p>The atmosphere in Kolkata is set to turn political as we are inching towards 2021. Most subjects
|
||
being discussed in Nabanna must be concerning the state elections to be held next year. It,
|
||
however,
|
||
serves as a temporary State Secretariat to West bengal. The original Secretariat is a 150-meter
|
||
long
|
||
building that covers the entire northern stretch of the iconic Lal Dighi pond in the downtown
|
||
area
|
||
of B.B.D. Bagh called the Writers’ building or just the Writers. The building with historical
|
||
importance has a distinct Greco-Roman style, with several statues of Greek gods as well as a
|
||
sculpture of Roman goddess Minerva that constitutes the prime attraction of this historical
|
||
heritage.
|
||
|
||
The building, as of today, has been under renovation since October 2013.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section3'>
|
||
<div class="head">
|
||
<div class="headline hl3">Writers Building, 2020, Calcutta, Painting Writers</div>
|
||
</div>
|
||
<p>The Writers Building is currently under renovation since 2013.
|
||
Next year sometime between April and June in 2021, about 294 constituencies will vote to decide
|
||
who
|
||
will become the next occupant of the chair in Writers Building</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section4'>
|
||
<div class="head">
|
||
<div class="headline hl3">Chief Minster Cabin, Calcutta, 2021, Painting the process
|
||
who will sit here-painting Chief Minister Cabin-Bridge to current occupant</div>
|
||
</div>
|
||
<p>Next year in sometime between March and June of 2021, about 67 million registered voters will
|
||
chose
|
||
294 representations to the state assembly. Writers Building has been under renovation since
|
||
2013. If
|
||
the renovation is finished by then, then 294 members of state legislative assembly will decide
|
||
the
|
||
next occupant of the Chief Minster's chamber in Writer's Building. The chair is currently
|
||
occupied
|
||
by a frail but fiestly lady who has never married Mamata Baneerjee.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section5'>
|
||
<div class="head">
|
||
<div class="headline hl3">Mamata Banerji, Calcutta, 2020, Profile about current
|
||
occupant-one interesting fact from her life to trigger - like being unmarried</div>
|
||
</div>
|
||
<p>As a young firebrand 29 year old, she caught attention of political pundits by defeating a
|
||
stalwart
|
||
of the communist party, Mr Somnath Chatterji in the central parliamentary elections in 1984 in
|
||
the
|
||
aftermath of sympathy wave generated due to assassination of Indira Gandhi representing Congress
|
||
Party.
|
||
|
||
|
||
Later in 1997 the veteran leaders in Congress Party could not hold her, and she split to form
|
||
her
|
||
own party Trinamool Congress which over time became the principle opposition party in West
|
||
Bengal,
|
||
and by 2011 she entered Writers Building as a Chief Minister. It was kind of a reversal of
|
||
fortunes
|
||
as she was dragged out forcefully from the same corridors in 1993 during a protest against
|
||
police
|
||
killings.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section6'>
|
||
<div class="head">
|
||
<div class="headline hl3">Hooghly-Origin of Calcutta, Calcutta, 1690, go back to
|
||
start how Calcutta was created in British times</div>
|
||
</div>
|
||
<p>Establishment of Calcutta, now called as Kolkata in modern independent India, was a long and
|
||
messy
|
||
ordeal for the British East India Company. Calcutta, as we know it, didn’t exist as a city back
|
||
then. Infact, what we know as Calcutta today was actually formed by
|
||
merging a cluster of three small villages; Sutanutti, Kalikata & Gobindpur. Job Charnok, who
|
||
was an
|
||
employee & an administrator in the British East India Company, is largely credited for the
|
||
formation
|
||
of the Calcutta.
|
||
|
||
The British East India company first came to Calcutta in the period of Ibrahim Khan( 1617-1624)
|
||
the
|
||
Subahdar Governor of Bengal at the time when Mughal Emperor Jahangir was ruling. They first
|
||
setup
|
||
factories in the city of Surat( 1620) and then later in Agra. Scouts were sent across the area
|
||
to
|
||
find out more suitable places to setup trade bases, though this plan was soon abandoned due to
|
||
logistical reasons. In January 1644, Emperor Jahangir’s daughter got severely burned in a
|
||
unfortunate incident when a British doctor named Gabriel Boughton was called upon for treatment.
|
||
Boughton was successfully able to treat the Emperor’s daughter for which as a reward the British
|
||
EIC
|
||
were allowed to establish a factory in Pipilli, Oddisha. This was the first time English ships
|
||
arrived at Eastern ports. Boughton’s medical services were availed again by the new Subedar of
|
||
Bengal, and in return the company was allowed to establish factories in Balasore, Odisha and
|
||
Hooghly, Bengal.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section7'>
|
||
<div class="head">
|
||
<div class="headline hl3">Aurangzeb, Dhaka, 1690, Explain State of Mughal affairs -
|
||
and how Bengal became independent country</div>
|
||
</div>
|
||
<p>Emperor Aurangzeb was the one sitting on throne during the major period of tension and conflict
|
||
between the Mughals and the British East India Company. The EIC wanted to establish a trade base
|
||
on
|
||
the eastern shore of India because of its strategic trading and military advantages.
|
||
|
||
|
||
In the year 1982 when Shaista Khan, the Governer of Bengal, was going to Delhi the EIC requested
|
||
him
|
||
to ask the Emperor to give permission to permanently trade in Bengal. Till this point they were
|
||
allowed to open factories in multiple places, but British couldn't be sure of their position.
|
||
The
|
||
Emperor was pleased to provide them with a Firman and this event was celebrated with 300 gun
|
||
shots
|
||
in Hoogly. Though one shouldn't be fooled by this happy start to the agreement, this was the
|
||
start
|
||
of a very long & painful cat and mouse chase between both.
|
||
|
||
|
||
Tensions started arising soon, as the Firman granted wasn't very clear. A conflict with a local
|
||
zamindar ended resulted in a putting a few Englishmen behind bars. Further the EIC requested the
|
||
state to give them a permission to build a fort on the mouth of Hooghly on the grounds of
|
||
protecting
|
||
their trade. But Shaista Khan didn't trust the intentions of British, he knew letting them build
|
||
a
|
||
fort right on the mouth of Hoogly would make the Mughals a little more vulnerable. Upon hearing
|
||
this
|
||
request he immediately turned it down and imposed another 3.5% tax on top of the agreed 3000
|
||
rupees,
|
||
breaking the Firni.
|
||
|
||
|
||
The company got very enraged with this situation and requested King James II in the year 1685 to
|
||
permit the use of force against the Mughals. The plan was to attack Chittagong, fortify it and
|
||
make
|
||
an alliance with the King of Arakan who was against the Mughals. Jon Carmock, who was in the
|
||
Madras
|
||
division at that time was told to join this expedition along with 400 soldiers from Madras
|
||
division.
|
||
The plan went awry though; because of high winds and bad weather conditions a lot of the ships
|
||
ended
|
||
up in Hoogly instead of Chittagong. Upon seeing large number of battle ships on the bank,
|
||
Governer
|
||
Shaista Khan got alarmed and immediately offered them truce. This didn't last long though, there
|
||
was
|
||
a dispute between British troops and the Mughal troops which led to an altercation between them
|
||
and
|
||
the latter beat the crap out of the former. The admiral then opened up fire on the city burning
|
||
down
|
||
500 houses; a property loss of about 30 lakh rupees. Upon Shaista Khan immediately sent troops
|
||
to
|
||
drive out the British from the area and capture all their factories and assets.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section8'>
|
||
<div class="head">
|
||
<div class="headline hl3">Murshid Quli Khan, Murshidabad, 1707-1717, Go upriver to
|
||
Murshidabad-Explain how the richest country in the world is born</div>
|
||
</div>
|
||
<p>The historical city of Murshidabad-the earstwhile nawabi capital –a city founded in the year 1704
|
||
by
|
||
Murshid Quli Khan, the Mughal diwan of Bengal. In 1704 Murshid Quli Khan had transferred the
|
||
capital
|
||
of Bengal from Dhaka to Murshidabad which he named the city after his name .The town is
|
||
situated on
|
||
the left bank of river Bhagirathi. Under the Nawabs Murshidabad’s glory reached to the highest
|
||
peak
|
||
in almost all arenas. As a trading centre Murshidabad became famous. Many scholars came here
|
||
,settled and mixed with the local people freely and hence there developed a cosmopolitan
|
||
culture. By
|
||
the middle of the 18th century Murshidabad became one of the greatest centres of culture and
|
||
education as the nawabs were the patrons of the learned.</p>
|
||
<p>In the Nawabi period Bengali literature also flourished .The learned men of this period made a
|
||
great
|
||
contribution in Urdu, Hindi, Persian literature. We ought to mention the Sufi literature of this
|
||
period .For the expansion of Islam Sufi litterateurs wrote books in Bengali to make the common
|
||
men
|
||
understand the Islamic knowledge and education. This proves that the state of education under
|
||
the
|
||
nawabs was excellent. During this period a different stream of Bengali language,Musalmani
|
||
Bangla,
|
||
arose which is a mix of Peresian, Urdu and Bengali languages. There was also a different form
|
||
of
|
||
literature ,mixed languages, which was popular among the lower section of the society. The elite
|
||
were less influenced by this literature. Persian was the official language of the era. The
|
||
Muslims
|
||
and Hindus both practiced this language which gave them the opportunity for a job in the
|
||
government
|
||
departments. Under the nawabs Murshidabad gained eminent position in the fields of education and
|
||
culture. Ghulam Hussain gives a long list of learned men, physicians, mathematicians and other
|
||
scholarly persons from this we can assume that educational institutions were there at
|
||
Murshidabad
|
||
during the period of review .Murshid Quli Khan established a Madrasha for the uplift of the
|
||
education. This institution , Kara Madrasha, as it was used as a guest house and was also
|
||
maintained
|
||
by the nawab.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section9'>
|
||
<div class="head">
|
||
<div class="headline hl3">Alivardi Khan, Murshidabad, 1717-1756, continuation of
|
||
Murshid Story-Alivardi was last strong man Britishers feared in Bengal</div>
|
||
</div>
|
||
<p>Alivardi Khan was the last strongman in Bengal whom the foreign powers feared. After his death,
|
||
in
|
||
1956, the dorrs to the paramount power in Bengal subah reopened.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section10'>
|
||
<div class="head">
|
||
<div class="headline hl3">Battle Theme- Sheikh Mujibur Rehman-Bangladesh, Bangladesh,
|
||
1971-1975, Creation of Bangladesh and Pakistan role in Bengal till Sheikh Mujibur Rehman was
|
||
assassinated-Bridge to his daugher Sheikh Haseena</div>
|
||
</div>
|
||
<p>While Britishers first entered into India from the west the true expansion and seeds of
|
||
colonialism
|
||
were sown in the east, particularly from Bengal. It was the year 1775 when a young British
|
||
governor
|
||
named Robert Clive must return to India as the governor of Fort St. David with a mission to
|
||
uproot
|
||
French administration from India. However, destiny must pull his attention towards Calcutta (now
|
||
Kolkata), a city that was rivaling Madras as a major trade centre in India. Clive knew the
|
||
strategic
|
||
importance and opportunities associated with Calcutta which, however, was under the
|
||
administration
|
||
of Siraj Al-Dawlah, the then Mughal viceroy of Bengal. For decades Britishers have had a mutual
|
||
understanding and cooperation with the Mughals. In order to strengthen British administration in
|
||
Bengal the new governor must make a decision. Clive decides to fortify the city. In response,
|
||
Clive
|
||
receives a brutal retaliation from Siraj Al-Dawlah who attacked and captured the fort.
|
||
|
||
|
||
Fall of Calcutta was a major setback for the British empire. It was perhaps Clive’s biggest
|
||
challenge to wrest Calcutta back from Mughals and to restore British supremacy in the region. An
|
||
angry and humiliated British governor had decided to establish British dominance in the region
|
||
and
|
||
restore British honor. An army of 900 Europeans and 1500 Indian soldiers marched towards
|
||
Calcutta
|
||
and recaptured it on January 2, 1757. The nawab is humiliated and forced to pay compensation and
|
||
let
|
||
the British fortify the city. Six months later, the humiliated nawab declares war against the
|
||
British popularly known as the Battle of Plassey. However, it was a one sided war where British
|
||
cannons unleashed havoc on the Siraj’s army. Clive overthroned Siraj and replaced him with Mīr
|
||
Jaʿfar, an elderly general secretly hostile to Sirāj al-Dawlah. With Mir Ja’far under his
|
||
influence,
|
||
Clive virtually became the master of the Bengal province. The event marked the advent of British
|
||
empire in India.
|
||
|
||
|
||
The same Britishers in 1905, however, partitioned Bengal which came to be known as the first
|
||
partition of Bengal by Lord Curzon, then viceroy of India. The partition was done for
|
||
territorial
|
||
reorganization of the Bengal Presidency. The partition separated the largely Muslim eastern
|
||
areas
|
||
from the largely Hindu western areas. The sentiments of a separate state on the basis of
|
||
religion,
|
||
faith and language were sown.
|
||
|
||
|
||
It's the year 1929, in a primary school in Gopalganj district, a young boy is protesting against
|
||
the
|
||
removal of his school principal for his hostile behavior against students. He organized a
|
||
student
|
||
union in order to achieve so. Such protests and rebels against any injustice were soon to become
|
||
a
|
||
part of his political career ahead for he was to become Bangabandhu Mujibur Rehman, the father
|
||
of
|
||
Bangladesh.
|
||
|
||
|
||
|
||
|
||
“Amar desh tomar desh..Bangladesh Bangladesh”
|
||
“Amar desh tomar desh..Bangladesh Bangladesh”
|
||
|
||
|
||
“My country. Your country. Bangladesh Bangladesh”. Surrounded by a huge mass gathering in Dhaka
|
||
in
|
||
the year 1971, angered by the negligence and hostility of the West Pakistan army and government
|
||
Mujibur Rehman, this time well popular and experienced is marching towards the podium to address
|
||
his
|
||
supporters. Since the inception of Pakistan in 1947, west Pakistan had a racist approach towards
|
||
the
|
||
people of East Pakistan. Pre-independance Mujib supported Jinnah's idea of partition of India on
|
||
the
|
||
basis of religion. But his hero turned villain when post independence Jinnah forced Urdu as the
|
||
only
|
||
official language of the whole of Pakistan which included its east counterpart as well. Mujib
|
||
protested.
|
||
|
||
|
||
Mujib’s rebellious behavior against west Pakistan marked a new phase in his political career
|
||
which
|
||
irked many in the west who were actually in power. From 1947 to 1971, jail became his second
|
||
home.
|
||
He opposed the biased behavior of his west counterparts towards Bangla people.
|
||
In 1958, General Ayub Khan declared martial law in the country. This period saw a significant
|
||
rise
|
||
of Islamic fundamentalism and degrading humanitarian rights and law especially towards the
|
||
people of
|
||
East Pakistan. After almost a decade of violence and distrust against Bangla people, in 1966,
|
||
Mujib
|
||
gave up the hope of equal status with western counterparts and demanded full autonomy for East
|
||
Pakistan in the form of Our Charter of Survival from West Pakistan. His demands, however, were
|
||
outrageous - a separate stock exchange and army to name few. The charter must have made
|
||
Pakistani
|
||
Generals either laugh or fume over it. Because Mujib was indirectly asking for a separate
|
||
country.
|
||
|
||
|
||
Nevertheless, Mujib was again arrested, invoking huge mass protests in East Pakistan for his
|
||
release
|
||
scaring the Pakistan army of a civil war. They agreed to release Mujib unconditionally. When
|
||
Mujib
|
||
reached Dhaka he was welcomed by people with great emotions and support for him. Surrounded
|
||
amongst
|
||
thousands of supporters he was conferred Bangabandhu or the friend of Bangla.
|
||
|
||
|
||
Finally, in 1970, the decade-long martial law was lifted and the general elections were
|
||
announced.
|
||
The results, however, invited more political unrest in Pakistan. Mujib’s Awami league came
|
||
victorious but were denied to form a government. Tensions between East Pakistan and West
|
||
Pakistan
|
||
rose to all time high. Under these situations Mujib retaliated by calling for full autonomy and
|
||
independence. He gave the slogan of “Joy Bangla” and declared independence. The ideology of
|
||
Bangladesh was formed.
|
||
|
||
|
||
He was again arrested, this time, on the charges of sedition. To suppress civil support for him
|
||
Pakistan army launched Operation Searchlight which even today is remembered as one of the most
|
||
barbaric and inhuman assaults on citizens. Thousands of people were killed, women raped,
|
||
children
|
||
crushed on the roads. Streets and houses of Dhaka were painted in red by the blood stains of
|
||
innocent Banglas. The Pakistan army men infiltrated each and every house of East Pakistanis and
|
||
launched fire. To this menace the people had no choice but to evade their homeland and flee to
|
||
more
|
||
secure neighbouring Indian states.
|
||
|
||
|
||
Thousands of East Pakistani crossed the border to find refuge in east India. It was the largest
|
||
humanitarian refugee crisis of that time. The sudden incursion of refugees alarmed people
|
||
sitting in
|
||
New Delhi. It was no more an internal issue of Pakistan. In the winters of 1971, Indira Gandhi,
|
||
the
|
||
then Prime Minister of India, declared war with Pakistan to support the local Bangla freedom
|
||
fighters called the Mukti Vahini. Field Marshal General Sam Manekshaw was given the
|
||
responsibility
|
||
to liberate Bangladesh from the brutalities of the west Pakistani regime. 13 days later about
|
||
91000
|
||
Pakistani soldiers surrendered to India. Yahiya Khan resigned. Under international pressure the
|
||
Pakistan government released Mujibur Rehman on 8th January, 1972. Bangladesh was born and
|
||
Mujibur
|
||
Rehman emerged as first the President and later the Prime Minister of Bangladesh.</p>
|
||
<p>The same Britishers in 1905, however, partitioned Bengal which came to be known as the first
|
||
partition of Bengal by Lord Curzon, then viceroy of India. The partition was done for
|
||
territorial
|
||
reorganization of the Bengal Presidency. The partition separated the largely Muslim eastern
|
||
areas
|
||
from the largely Hindu western areas. The sentiments of a separate state on the basis of
|
||
religion,
|
||
faith and language were sown.
|
||
|
||
|
||
It's the year 1929, in a primary school in Gopalganj district, a young boy is protesting against
|
||
the
|
||
removal of his school principal for his hostile behavior against students. He organized a
|
||
student
|
||
union in order to achieve so. Such protests and rebels against any injustice were soon to become
|
||
a
|
||
part of his political career ahead for he was to become Bangabandhu Mujibur Rehman, the father
|
||
of
|
||
Bangladesh.</p>
|
||
<p>The election result of 1971 was not honoured, leading to a civil war on March 26, and Sheikh
|
||
Mujib
|
||
declared Bangladesh’s independence from Pakistan . The declaration coincided with a ruthless
|
||
show of
|
||
strength by the Pakistani military, in which tanks rolled out on the streets of Dhaka and
|
||
several
|
||
students and intellectuals were killed.
|
||
A woman only 13, had already learned to fear people around her. She saw her father being to
|
||
death.One day in late afternoon people barged into her house. They snatched her two little
|
||
brothers,
|
||
tied them to a tree and beat them.She ran out of the front door , but the men caught her. They
|
||
tethered her arms to two trees. They ripped off her earrings and bracelets, and stripped off her
|
||
clothes.
|
||
|
||
|
||
Between 300,000 and 3,000,000 people were killed and around 200,000 to 400,000 women were raped.
|
||
|
||
|
||
India under then Prime Minister Indira Gandhi provided full support to Rahman and Bangladesh’s
|
||
independence movement, resulting in the creation of a sovereign government at Dhaka in January
|
||
1971.
|
||
</p>
|
||
<p>“Amar desh tomar desh..Bangladesh Bangladesh”
|
||
“Amar desh tomar desh..Bangladesh Bangladesh”
|
||
|
||
|
||
“My country. Your country. Bangladesh Bangladesh”. Surrounded by a huge mass gathering in Dhaka
|
||
in
|
||
the year 1971, angered by the negligence and hostility of the West Pakistan army and government
|
||
Mujibur Rehman, this time well popular and experienced, is marching towards the podium to
|
||
address
|
||
his supporters. Since the inception of Pakistan in 1947, west Pakistan had a racist approach
|
||
towards
|
||
the people of East Pakistan. Pre-independance Mujib supported Jinnah's idea of partition of
|
||
India on
|
||
the basis of religion. But soon, his national hero and ideal turned villain when post
|
||
independence
|
||
Jinnah announced Urdu as the only official language of the whole of Pakistan which included its
|
||
east
|
||
counterpart as well. Mujib protested.</p>
|
||
<p>One of the great grandsons of Mir Zafar was Iskandar Mirza, who assumed the first Presidency of
|
||
Pakistan after giving his country a constitution in 1956, full 9 years after Pakistan's
|
||
Independence. A proud and stout Bengali, he found it difficult to get along with the Prime
|
||
Ministers he appointed. Like a musical game of chairs, he changed four Prime Ministers in his
|
||
Presidency from March of 1956 to October of 1958. His third Prime Minister was Huseyn Suhrawady,
|
||
an
|
||
influential Bengali Muslim freedom fighter who also once held Prime Ministership of united
|
||
Bengal
|
||
province during Britsh times.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section11'>
|
||
<div class="head">
|
||
<div class="headline hl3">Sheikh Haseena, Dhaka-Bangladesh, 2020, Regeneration to be
|
||
the fast growing region in the world-Bridge to Mamata</div>
|
||
</div>
|
||
<p>Sheikh Hasina was sworn in as prime minister of Bangladeh in January 2009. In 2017, in the midst
|
||
of
|
||
Hasina's premiership, more than 700,000 Rohingya arrived in Bangladesh, fleeing genocide in
|
||
neighbouring Myanmar .
|
||
The Bangladesh government provided refuge and assistance, however, it did not grant refugee
|
||
status
|
||
and worked to repatriate the Rohingyas on a voluntary basis.
|
||
Hasina's government received praise both internationally and domestically for helping the
|
||
Rohingya.
|
||
In past years Bangladesh Prime Minister Sheikh Hasina is in discussion with West Bengal Chief
|
||
Minister Mamata Banerjee to discuss a range of bilateral issues.
|
||
The Teesta river water sharing issue was discussed by the two leaders and New Delhi and Dhaka
|
||
both
|
||
are keen to seal an agreement on it since September 2011.
|
||
Sheikh Hasina has time and again acknowledged the friendly and cordial relation between the two
|
||
countries to continue and flourish.</p>
|
||
<p>Under Hasina, Bangladesh has made significant strides towards educating girls and giving women a
|
||
greater voice, both in the household and the public sphere whose efforts have translated into
|
||
improvements in children’s health and education, such that Bangladeshis’ average life expectancy
|
||
is
|
||
now 72 years, compared to 68 for Indians and 66 for Pakistanis.
|
||
The Bangladesh government deserves credit for supporting grass-roots initiatives in economic
|
||
inclusion which includes Bangladeshi adults with bank accounts, 34.1% made digital transactions
|
||
in
|
||
2017, compared to an average rate of 27.8% for South Asia.
|
||
Hasina’s government has offered a better environment for manufacturing firms to achieve
|
||
economies of
|
||
scale and create a large number of jobs, and though Bangladesh still needs much stronger
|
||
regulation
|
||
to protect workers from occupational hazards, the absence of which curtails labour-market
|
||
flexibility has been a boon for job creation and manufacturing success.</p>
|
||
<p>Mamta Banerjee has also extended her support to Sheikh Hasina by protesting Modi's govt on CAB
|
||
and
|
||
NRC Bill.
|
||
Mamta's stand on the above issue explains why the bilateral relation between Hasina and her is
|
||
going
|
||
to last long.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section12'>
|
||
<div class="head">
|
||
<div class="headline hl3">Mamata Banerji, Calcutta, 2020, Another aspect of her life-
|
||
a short event - Bridge to her political predecessor system Communism and Jyoti Basu</div>
|
||
</div>
|
||
<p>The dynamics of democratic politics in West Bengal, a major state of eastern India had taken its
|
||
own
|
||
shape despite some aberrations due to the politics of violence in the sixties. Whenever there
|
||
has
|
||
been any communication gap between the ruling party and the people, as it occurred during
|
||
1964-66
|
||
and again during 1974-76, the electorate democratically registered its protest at the next
|
||
available
|
||
opportunity by rejecting the imperious attitude of the ruling party .Such a gap has been very
|
||
much
|
||
bridged by the direct and personal type of leadership adopted by the left politicians
|
||
ideological
|
||
slant of the political process in West Bengal has, in general, prevented the growth of regional
|
||
patriotism or politics Of parochialism, except for the minor political impact left by Ghisingh
|
||
and
|
||
his GNLF movement. It is one of the sings of politics in West Bengal.</p>
|
||
<p>The elected Leftist coalition since 1977 (often described as sound ‘political stability’), but
|
||
also
|
||
because of its widely acknowledged successes in fertility transition, execution of
|
||
re-distributive
|
||
land reform and political decentralization programmes. Ironically, yet, the state, in almost all
|
||
comparative assessments of social, human and infrastructural developments occupies a lagged
|
||
position
|
||
vis-à-vis many other states, especially in the south and even against all-India records. A
|
||
well-disciplined grassroots political mobilization network, and the machinery of the Left Front
|
||
parties, have been highly instrumental for comparatively fast declines of fertility and
|
||
population
|
||
growth and for lasting political stability in an otherwise ‘dilatory’ development regime.
|
||
However, a
|
||
government geared to ensuring mass electoral support overwhelmingly via a grassroots
|
||
mobilization
|
||
network but, with a relative neglect of social movements, economic infrastructure and human
|
||
development, is likely to suffer adverse consequences in the longer term.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section13'>
|
||
<div class="head">
|
||
<div class="headline hl3">Jyoti Basu-Communism, West Bengal, 1977-2000, Impact of
|
||
Communism in Bengal-Bridge to Cornwallis Permanent Settlement</div>
|
||
</div>
|
||
<p>The ripple effects of Russian revolution in 1917 reached British India that was fighting against
|
||
a
|
||
similar “class” ideology. Consequently, in 1925 the foundation of communism was set in Kanpur.
|
||
However, due to weak leadership and dominance of Gandhian philosophy, communism in India was
|
||
finding
|
||
it difficult to sustain its roots in Indian politics. 20 years later, a young Jyoti Basu was
|
||
attending the lectures of Harold Laski, a prominent member of the British Labour Party turned
|
||
Marxist. This small event was going to change everything in Basu’s ideological philosophy as
|
||
well as
|
||
Indian politics in future. In the years that followed, Basu’s activities in politics
|
||
significantly
|
||
increased and, in 1939, selected as the General Secretary of the London Majlis he came into
|
||
contact
|
||
with prominent Indian political leaders like Nehru and Bose. He returned to India in 1940 and
|
||
joined
|
||
the Communist Party of India.</p>
|
||
<p>At the time of independence, all of India from north to south was governed popularly by one
|
||
party,
|
||
the Indian National Congress. Bengal was no different. Basu realized that if he had to establish
|
||
communist ideology in Bengal he must establish communications with the lower sections of the
|
||
society
|
||
especially with farmers and peasants. He got this opportunity in the form of Te-Bhag movement
|
||
under
|
||
which small farmers united against the zamindars or the landlords with the demand to only give
|
||
one-third of the land tax. The movement was heavily supported by CPI led by Jyoti Basu. Over the
|
||
years he established various worker unions and also played a pivotal role in uniting various
|
||
communist parties (like the Forward Bloc, the Socialist Replublican party and the United
|
||
Socialist
|
||
party) all across India to build an anti-Congress front. However, after the Sino-India war in
|
||
1962
|
||
CPI split into two parts - CPI(Marxist) and CPI(Maoist) following ideological differences.</p>
|
||
<p>The influence of CPI(M) in Bengal gave a stiff competition to Congress and in 1977 when CPI(M)
|
||
came
|
||
to power Jyoti Basu served a (then) record 23 years as the Chief Minister of Bengal. The prime
|
||
reason for such a long tenure of communism in Bengal can be found in its basic ideology which
|
||
stands
|
||
up for the rights of workers class. Under his long 33 years tenure Jyoti basu made several
|
||
landmark
|
||
reforms for the deprived class. Empowering villages from grassroot level by empowering the
|
||
Panchayat
|
||
System, boosting labour reforms and land reforms were among few of them which tightened his
|
||
political grip on the socially backward class. His secular personality further improved his
|
||
image
|
||
amongst people of all backgrounds and race.</p>
|
||
<p>Operation Barga was a land reform movement throughout rural West Bengal for recording the names
|
||
of
|
||
sharecroppers (bargadars) while avoiding the time-consuming method of recording through the
|
||
settlement machinery. It bestowed on the bargadars, the legal protection against eviction by the
|
||
landlords, and entitled them to the due share of the produce. Operation Barga was launched in
|
||
1978
|
||
and concluded by the mid-1980s.
|
||
|
||
|
||
Introduced in 1978, and given legal backing in 1979 and 1980, Operation Barga became a popular
|
||
but
|
||
controversial measure for land reforms. The ultimate aim of these land reforms was to facilitate
|
||
the
|
||
conversion of the state's bargadars into landowners. To date, Op Barga has recorded the names of
|
||
approximately 1.5 million bargadars. Since then, it has been marked as one of the more
|
||
successful
|
||
land reforms programs in India.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section14'>
|
||
<div class="head">
|
||
<div class="headline hl3">Cornwallis, Bengal, 1790-1800, Permanent Settlement System
|
||
and how it impacted West Bengal</div>
|
||
</div>
|
||
<p>From time Immemorial the State in India has claimed a certain proportion of the produce of every
|
||
bigha or acre of land. The laws of Manu declare that a king is entitled to an eighth, a sixth,
|
||
or a
|
||
twelfth of the gross product. according to the difference of the soil and the labours necessary
|
||
to
|
||
cultivate it: in times of emergency, even a fourth. Alauddin Khilji levied a tax equal to half
|
||
the
|
||
gross produce of the lands in his kingdom. Under Sher Shah and Akbar the share of the State was
|
||
a
|
||
third; thereafter the demand increased, reaching up to a half.
|
||
|
||
|
||
Land Revenue was the chief source of public income in India till 1916.</p>
|
||
<p>The word Zemindar means literally a land-holder. In Bengal the Mughal conquerors used the term to
|
||
denote those persons, drawn from almost every level of society, whom they employed to collect
|
||
the
|
||
revenue from the land. In lieu of salaries, these zemindars were given grants of land free from
|
||
revenue, and allowed to retain a specified portion of the collections, to levy transit duties,
|
||
and
|
||
to dispose of the waste lands. In origin, ilierefore, the zemindari tenure in Bengal was an
|
||
official
|
||
one. Its sole foundation was the sanad, or charter of appointment. Its essential feature was the
|
||
collection and transmission of public revenue. In the heyday of Mughal ride sanads were often
|
||
revoked on failure to render prompt payments. Some zemindars, it is true, succeeded in
|
||
stabilising
|
||
their position, and claimed a customary and prescriptive right to collect the revenue. But this
|
||
in
|
||
itself did not divest the zemindari tenure of its official character; for under the Mughals all
|
||
offices tended to become hereditary.</p>
|
||
<p>Ijarah System was introduced in Mughal empire under Jahandhar Shah by Zulfiqar Khan.
|
||
|
||
|
||
Under this system the government gave the right to collect tax revenue on production to a third
|
||
party (middle man) and in return got an assured sum from the third party. The third party could
|
||
levy
|
||
any percentage of tax on the people.
|
||
|
||
|
||
This was different from the earlier system as incase of the older system tax was collected at a
|
||
fixed rate (fixed by the state) and could not be arbitrarily changed by individuals/tax
|
||
officials.
|
||
|
||
|
||
Ijarah system exposed the population to a tyrranous tax system because they were left at the
|
||
mercy
|
||
of the middlemen who continuously exploited them.</p>
|
||
<p>Revenue system of Mauryan Empire was devised by the Minister of Chandragupta Maurya, Kautilya.
|
||
The
|
||
principal items of revenue in town and country have been listed by Kautilya while describing the
|
||
duties of the Samaharta (collector-general).
|
||
|
||
|
||
The main heads of revenue were: A share of land output. Other dues and cesses levied on land
|
||
including a water-rate. The water rate varied according to the nature of the land and crop and
|
||
tax
|
||
on houses in towns. Income from crown lands, from forests, from mines and manufactures, some of
|
||
them
|
||
monopolies like salt, undertaken by government Customs at the frontiers and tariffs, tolls and
|
||
ferry dues in the interior levied on merchandise in transport Profits of coinage and gains from
|
||
trade operations carried on by the government Fees for licenses of various kinds to be taken
|
||
out by
|
||
artisans, craftsmen, professionals and traders Fines levied in law courts and Miscellaneous
|
||
receipts like presents like transferred possession.
|
||
In
|
||
emergency 'benevolences' were resorted to and the rich had to pay considerable amounts to the
|
||
state
|
||
under. Mauryas introduced images with a view to gain gold. Granting exemptions from payment of
|
||
revenue was common particularly land-revenue to Brahmins and religious institutions and to
|
||
state-officials. A register of such remissions and assignments was maintained.
|
||
|
||
Industries were taxed, using a vast mix of techniques, all of which
|
||
were derived from the original land revenue tax system. This entire system was the creation of
|
||
Kautilya, the prime minister of Chandragupta Maurya. As the taxation system developed the
|
||
government
|
||
had money to further develop the kingdom. The government also introduced state owned farms where
|
||
the
|
||
king owned the land and his subjects were employed for its cultivation. The government had the
|
||
resources to organize the procedure and once the land was ready it would employ the cultivator
|
||
to
|
||
work on the land. The Mauryas had developed
|
||
extensive trade relations. There were many foreign traders who took up residence in Mauryan
|
||
cities.
|
||
There were a variety of goods being exported thereby generating revenue. The exports were items
|
||
of
|
||
luxury like fine muslin cloth. The development of trade became profitable for the empire and
|
||
eventually a separate department looked after trade and commerce. Land revenue of Mauryan
|
||
Empire
|
||
Land revenues from the rural areas were appropriated in the form of crown
|
||
lands (sita), land revenue (bhaga) from cultivators, taxes on orchards, ferry charges and so
|
||
on.
|
||
|
||
|
||
Different types of taxes that were imposed on rural population are:
|
||
(i) Bhaga - It was levied at the rate of one-fourth to one-sixth. Sharecroppers and other
|
||
agricultural support by the state had to pay half of the produce to the state. \(ii)
|
||
Pindikara -
|
||
It was a tax levied on groups of villages and paid by farmers. This was customary. Often the
|
||
villages were supposed to supply provisions to the royal army when they pass through their
|
||
respective territories.
|
||
(iii) Hiranya - It was a tax paid in cash.
|
||
(iv) Bali - This was popular in the Vedic times and continued under the Mauryas.
|
||
(v) Pranaya - This literally means a gift of affection. It was levied imposed by the state
|
||
during
|
||
emergency. This was first mentioned by Panini, it was elaborated upon for the first time in
|
||
Arthashastra. Pranaya amounted to one-third or one-fourth of the produce according to the nature
|
||
of
|
||
the soil.
|
||
|
||
|
||
Export-Import Tax:- Import tax was 20 per cent, but there
|
||
is no
|
||
definite knowledge about the export tax. Import tax was called prabeshya, while export tax was
|
||
called nishkramya. Sales Tax:- The Sulkadhyaksha collected toll tax on every
|
||
item
|
||
before it was sold or purchased. There were three rates of toll tax or sale tax: 9.5 per cent on
|
||
iteIns sold on the basis of calculation; 5 per cent on the iteIns sold on the basis of
|
||
measurement;
|
||
and 6.5 per cent on items sold on the basis of weight. Income Tax:-
|
||
The following taxes were imposed on the sources of income in cities: (i) wine manufacturing tax;
|
||
(ii) salt manufacturing tax; (iii) taxes on ghee, oil and edible oil; (iv) taxes on animal
|
||
slaughterers; (v) taxes on artisans and artists; (vi) taxes on gamblers and gambling houses;
|
||
(vii)
|
||
taxes on prostitution, (viii) taxes on the income of temples; and (ix) taxes on additional
|
||
incomes
|
||
of the wage earners.</p>
|
||
<p>Taxation in Gupta Empire
|
||
|
||
The Economy during the Gupta period was moving from 18 sources or types of Taxes levied on the
|
||
peolple; the greater part of it was spent on works of public welfare. Land revenue was the main
|
||
source was generally fixed at one-sixth of the produce.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section15'>
|
||
<div class="head">
|
||
<div class="headline hl3">William Bentick, Bengal, 1828-1835, Bentick social
|
||
reforms</div>
|
||
</div>
|
||
<p>Lord William Bentinck was born in London, the second son of the 3rd Duke of Portland. He was
|
||
educated
|
||
at Rev. Dr Samuel Goodenough's school in Ealing, before moving on to Westminster School. After
|
||
completing his education, he embarked on a military career, moving swiftly up the ranks on
|
||
account
|
||
of his family's influence and his ability to purchase commissions.
|
||
|
||
|
||
In 1799, Lord William went to Italy, where he served for two years as liaison officer with the
|
||
Austro-Russian forces. In 1801 he embarked for Egypt to command the cavalry, but saw no active
|
||
service. His star continued to rise with his appointment, in 1803, as Governor of Madras. The
|
||
appointment had been secured largely by the lobbying of his father, but was not popular with the
|
||
East India Company and the Board of Control. Lord William had a number of significant
|
||
achievements
|
||
in India, but his work was largely overshadowed by the mutiny at Vellore in July 1806. He left
|
||
his
|
||
post in 1807.
|
||
|
||
|
||
In 1808 he served in the Battle of Corunna and by 1811 had been appointed envoy to the Court of
|
||
the
|
||
Two Sicilies. From here, he promoted plans for an independent and united Italy, until he was
|
||
recalled to Britain in 1815.
|
||
|
||
|
||
After a period at home in Britain when he served as a commissioner for drainage and navigation
|
||
in
|
||
the fens, he returned to India in 1828 as Governor General - a post which he then held until
|
||
1835.
|
||
In this role he undertook sweeping social, economic and political reforms which, it has been
|
||
argued,
|
||
laid the foundations for modern India.</p>
|
||
<p>He refused reappointment to the governorship of Madras in 1819, waiting to attain his real
|
||
ambition—the appointment as governor-general of Bengal, which came in 1827. Bentinck’s immediate
|
||
instructions were to rescue India from its financial difficulties; at this time the government
|
||
in
|
||
India operated on an annual deficit of about £1.5 million. Bentinck soon succeeded in turning
|
||
the
|
||
deficit into a surplus of about the same amount. The result of his efforts was the renewal of
|
||
the
|
||
East India Company’s government by the Charter Act of 1833, whereby Bentinck became the first
|
||
governor-general of India. He next turned to personnel reforms, which included making more
|
||
administrative and judicial positions available to Indians and improving the salaries and status
|
||
of
|
||
Indian judges. Bentinck also made English, instead of Persian, the language of the higher courts
|
||
and
|
||
of higher education and arranged for financial aid to colleges, which were to be adapted to the
|
||
Western models.
|
||
|
||
|
||
Bentinck showed great courage and humanity by his decision to abolish suttee (sati), the Hindu
|
||
custom of burning widows alive with the corpses of their husbands. Previous governors-general
|
||
had
|
||
shrunk from prohibiting the custom as an interference in religion and one particularly likely to
|
||
upset the Indian army, but Bentinck cut through these hesitations without facing much open
|
||
opposition. He was also responsible for the measures taken to suppress the murder of unwanted
|
||
children, human sacrifice, and the thags—bands of robbers, bound together by oaths and ritual,
|
||
who
|
||
murdered unsuspecting travelers in the name of the goddess Kālī. Flogging in the Indian army was
|
||
also abolished, long before it ended in the British army.</p>
|
||
<p>At his tenure Governor General of Bengal became Governor General of India. He was responsible
|
||
for
|
||
introduction of English as a language of instruction in India. He reformed the finances, opened
|
||
up
|
||
judicial posts to Indians, and suppressed such practices as suttee, or widow burning, and
|
||
thuggee,
|
||
or ritual murder by robber gangs</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section16'>
|
||
<div class="head">
|
||
<div class="headline hl3">Lord Dalhousie, Bengal, 1857, events leading to first war
|
||
of independence</div>
|
||
</div>
|
||
<p>Lord Canning 1858 to 1862
|
||
|
||
|
||
The Government of India Act, 1858
|
||
System of Budget introduced
|
||
Formation of Imperial Civil Services
|
||
Indigo Revolt in Bengal in 1859-60
|
||
Enactment of Indian Penal Code in 1860
|
||
Establishment of Archaeological Survey of India</p>
|
||
<p>He reorganised Calcutta Secretariat and appointed a Lieutenant Governor for Bengal to relieve the
|
||
administrative burden of the Governor General. He laid down the main lines of development of
|
||
railway
|
||
system in India. Telegraphs were setup and the postal system was reformed.</p>
|
||
<p>The Santhal rebellion (sometimes referred to as the Sonthal rebellion), commonly known as Santhal
|
||
Hool, was a rebellion in present-day Jharkhand, in eastern India against both the British
|
||
colonial
|
||
authority and zamindari system by the Santhal people. It started on June 30, 1855 and on
|
||
November
|
||
10, 1855 martial law was proclaimed which lasted until January 3, 1856 when martial law was
|
||
suspended and the movement was brutally ended by troops loyal to the British. The rebellion was
|
||
led
|
||
by the four Murmu Brothers - Sidhu, Kanhu, Chand and Bhairav</p>
|
||
<p>The insurrection of the Kolarian Santals of Bengal against the extortions of landlords and
|
||
moneylenders had been severely repressed, but the causes of the insurrection had still to be
|
||
reviewed and a remedy provided. By removing the tract of country from local rule, enforcing the
|
||
residence of British officers there, and employing the Santal headmen in a local police, he
|
||
created
|
||
a system of administration which proved successful in maintaining order.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section17'>
|
||
<div class="head">
|
||
<div class="headline hl3">Lord Curzon, Bengal, 1905, partition of Bengal</div>
|
||
</div>
|
||
<p>Partition of Bengal, (1905), division of Bengal carried out by the British viceroy in India, Lord
|
||
Curzon, despite strong Indian nationalist opposition. It began a transformation of the Indian
|
||
National Congress from a middle-class pressure group into a nationwide mass movement.</p>
|
||
<p>Bengal, Bihar, and Orissa had formed a single province of British India since 1765. By 1900 the
|
||
province had grown too large to handle under a single administration. East Bengal, because of
|
||
isolation and poor communications, had been neglected in favour of west Bengal and Bihar. Curzon
|
||
chose one of several schemes for partition: to unite Assam, which had been a part of the
|
||
province
|
||
until 1874, with 15 districts of east Bengal and thus form a new province with a population of
|
||
31
|
||
million. The capital was Dacca (now Dhaka, Bangl.), and the people were mainly Muslim.
|
||
|
||
|
||
The Hindus of west Bengal, who controlled most of Bengal’s commerce and professional and rural
|
||
life,
|
||
complained that the Bengali nation would be split in two, making them a minority in a province
|
||
including the whole of Bihar and Orissa. They regarded the partition as an attempt to strangle
|
||
nationalism in Bengal, where it was more developed than elsewhere. Agitation against the
|
||
partition
|
||
included mass meetings, rural unrest, and a swadeshi (native) movement to boycott the import of
|
||
British goods. The partition was carried through despite the agitation, and the extreme
|
||
opposition
|
||
went underground to form a terrorist movement.</p>
|
||
<p>In 1898 it was announced that he would succeed Lord Elgin as viceroy of India, and, in September
|
||
of
|
||
that year, he was created Baron Curzon of Kedleston. He was the youngest viceroy of India in
|
||
history, and he cherished the prospect of it, for it was an office filled with the pomp and
|
||
ceremony
|
||
he loved. India was the most-treasured jewel in Queen Victoria’s crown, and, after the pageantry
|
||
of
|
||
his arrival in Calcutta on January 3, 1899, Curzon wrote: “I suddenly saw what had come into my
|
||
hands, and what prodigies of energy and inspiration would be needed on my part to guide them.”
|
||
He
|
||
demanded obedience and a bent knee from the rajas, maharajas, and provincial governors he now
|
||
ruled,
|
||
but there was no doubt of his sense of mission toward the Indian people. He initiated
|
||
commissions of
|
||
inquiry into education, police, and civil services; he reduced taxes; he ordered immediate
|
||
punishment of any Briton (including members of the army) who ill-treated Indian nationals. In
|
||
external affairs he paid special attention to India’s frontiers, toured the Persian Gulf, and
|
||
sent a
|
||
successful mission to Tibet to frustrate Russian ambitions there. On his return from what he
|
||
called
|
||
a “triumphal” tour of the Indian provinces, he ordered the restoration of the Taj Mahal, which
|
||
was
|
||
decaying, and thereafter took a personal interest in India’s artistic and cultural heritage.</p>
|
||
<p>At the end of his first five years in India, his successes were recognized by the government at
|
||
home
|
||
by a renewal of his term, but, in fact, the period of blazing glory was over and now came
|
||
political
|
||
tragedy. At Curzon’s personal request, the job of commander in chief of the Indian army and
|
||
military
|
||
member of the viceroy’s cabinet had been given to England’s military hero of the day, Lord
|
||
Kitchener
|
||
of Khartoum. Curzon believed that to have such an illustrious soldier on his staff would
|
||
embellish
|
||
his own image, though his friends in England warned him repeatedly that Kitchener was, in Lord
|
||
Esher’s words, “an uncouth and ruthless man.” It was a clash of personalities, and the two were
|
||
soon
|
||
unscrupulously intriguing against each other. A final confrontation between the two men, which
|
||
ended
|
||
in Curzon’s bursting into tears, brought matters to a climax. Curzon, confident that the
|
||
government
|
||
would take his part, cabled that either his views must be accepted or he would go. On the
|
||
morning of
|
||
August 16, 1905, he received a cable from King Edward VII telling him that his resignation had
|
||
been
|
||
accepted. He delayed his return to England, and, by the time he was once again in London, the
|
||
Tories
|
||
were out of office, and his Indian achievements had been forgotten. He was not even given the
|
||
earldom usually awarded to retiring viceroys.</p>
|
||
<p>In 1911, the year that the capital was shifted from Calcutta (now Kolkata) to Delhi, east and
|
||
west
|
||
Bengal were reunited; Assam again became a chief commissionership, while Bihar and Orissa were
|
||
separated to form a new province. The aim was to combine appeasement of Bengali sentiment with
|
||
administrative convenience. This end was achieved for a time, but the Bengali Muslims, having
|
||
benefitted from partition, were angry and disappointed. This resentment remained throughout the
|
||
rest
|
||
of the British period. The final division of Bengal at the partitioning of the subcontinent in
|
||
1947,
|
||
which split Bengal into India in the west and East Pakistan (later Bangladesh) in the east, was
|
||
accompanied by intense violence.</p>
|
||
</div>
|
||
</div>
|
||
<div class="columns">
|
||
<div class="column" id='section18'>
|
||
<div class="head">
|
||
<div class="headline hl3">Writers Building-Bidhan Chandra Roy, Calcutta, 1950,
|
||
explain his contribution in governing West Bengal</div>
|
||
</div>
|
||
<p>B.C. Roy was a great planner for the improvement of education and culture of West Bengal. Due to
|
||
the
|
||
efforts of B.C. Roy Viswa Bharati, established by Rabindra Nath Tagore in 1921 was elevated as
|
||
Viswa
|
||
Bharati University under the Governance of the Government of India in 1951. It may be noted here
|
||
that the planning for the establishing of the above mentioned Kalyani University was initiated
|
||
by
|
||
B.C. Roy in 1954. It was B.C. Roy who took the special drive for the passing of the Jadavpur
|
||
University Bill in the Legislative Assembly in 1955. As a President of the National Education
|
||
Council, B.C. Roy himself moved the Jadavpur University Bill in the month of September 1955.
|
||
Jadavpur University was occasionally opened by Dr. Sarvapally Radhakrishan, the then Vice-
|
||
President
|
||
of India on 18th March 1956 for opening up a new era in the history of the higher education in
|
||
India
|
||
particularly in West Bengal. ‘Academy of Dance, Drama and Music’ was established in 1955. As a
|
||
result of B.C. Roy’s vision for commemorating to Gurudev Rabindra Nath Tagore a plan was
|
||
formulated
|
||
for establishing Rabindra Bharati University in 1959. He was also the founder President of
|
||
Indian
|
||
Institute of Social Welfare and Business Management in Calcutta.</p>
|
||
<p>Maulana Abul Kalam Azad, the first Education Minister of free India, was also the personal friend
|
||
of
|
||
Bidhan Chandra Roy, the Chief Minister of West Bengal. In course of discussion between the two
|
||
the
|
||
former approached the later to locate a site for setting up one of the five regional engineering
|
||
colleges sometime in the year of 1950-51. The Chief Minister B.C. Roy accepted the offer at once
|
||
and
|
||
selected Hijli at Kharagpur in the district of Midnapur which is a distance of 100 miles from
|
||
Calcutta for the purpose of setting up engineering college. Due to the insistence of Maulana
|
||
Abul
|
||
Kalam Azad prior to the approval of I.I.T. Kharagpur B.C. Roy agreed to take overall charge of
|
||
the
|
||
Institute as its Chairman in his personal capacity. With the help of his personal sagacity and
|
||
indomitable spirit for doing something for the betterment of the state, B.C. Roy selected Jnan
|
||
Chandra Ghosh, an eminent scholar and scientist as its first Director.</p>
|
||
<p>His Bihar Bengal merger scheme with Srikrishna Sinha failed</p>
|
||
<p>One of the main causes behind the shortage of food was the partition of the country which gave
|
||
almost
|
||
20 million acres of excellent rice-growing paddy fields to the East Pakistan and thereby rice
|
||
became
|
||
comparatively unavailable and more expensive in West Bengal. In addition the influx of lakhs of
|
||
refugees from East Pakistan to West Bengal, diversion of 200,000 acres of rice growing lands to
|
||
jute
|
||
cultivation and damage of crops due to natural calamities - collectively resulted in a yearly
|
||
deficit of 200,000 tons food grain in the state.</p>
|
||
<p>Zamindari Abolition Bill There were some important Bills relating to the land system passed
|
||
during
|
||
the chief ministership of B.C. Roy. In the ‘Bargaders Acts’ of 1950 some provisions were there
|
||
to
|
||
protect the share croppers. According to the Acts a ‘share croppers conflict –removal board’ was
|
||
set
|
||
up. Not only that ‘Zamindari Accusation Acts, and ‘Land Reforms Acts’ were passed in 1953 and
|
||
1955
|
||
respectively. According to the ‘Zamindari Accusation Acts of 1953, the permanent settlement and
|
||
Zamindari systems were declared null and void. There were provisions to give compensations to
|
||
the
|
||
Zamindars. However, on and from 1st Baishak, 1362 (Bengali Callender) i.e.1955 155 the
|
||
accusation
|
||
of Zamindari was started by the Government of West Bengal. The shares of the share-croppers were
|
||
fixed 60% instead of tebhaga i.e. 2/3. It was circulated that the owner of the vested lands
|
||
would be
|
||
allowed to retain 25 acres of cultivable lands and 15 acres of housing as well as garden lands.
|
||
Measures were taken to distribute the surplus lands among the share croppers and landless
|
||
peasants.
|
||
The abolition of Zamindari system by the West Bengal Government in 1953, marked the beginning of
|
||
one
|
||
of the most revolutionary measures in the history of the land reforms in the country to stop
|
||
exploitation of peasantry by the landlords.173With the passage of the Bill the power and
|
||
prestige of
|
||
the landed aristocracy began to decline steadily. The Zamindars started selling their palatial
|
||
buildings which they could no longer maintain. In that connection, the State Government was
|
||
advised
|
||
by the Central Government to examine the cases and purchase such properties for utilization of
|
||
nation- building purpose.</p>
|
||
<p>As Chief Minister of West Bengal, he laid the foundation of five cities, namely, Durgapur,
|
||
Kalyani,
|
||
Bidhannagar, Ashokenagar and Habra.</p>
|
||
<p>In the obituary to Dr Bidhan Chandra Roy, the British Medical Journal wrote, “… at his
|
||
professional
|
||
zenith, he may have had the largest consulting practice in the world, news of his visit to a
|
||
city or
|
||
even railway station bringing forth hordes of would-be patients.”</p>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<span style="padding-left:10rem;cursor:pointer;border: black solid 2px;padding: 6px;margin-left: 10rem;" id="column${
|
||
i + 1
|
||
}_thumbs" class="fa fa-thumbs-o-up"
|
||
onclick="clickProcess(this)"> Upvote
|
||
<!--<i class="fa fa-thumbs-o-up" aria-hidden="true" style="padding-left:1rem;cursor:pointer;"></i>-->
|
||
<span style="font-weight:bold;padding-left: 5px;" id="num"> </span></span>
|
||
<!-- <div><img src="https://cdn-icons-png.flaticon.com/512/25/25423.png" alt="" style="height: 25px;margin-left: 30%;">Upvote</div>-->
|
||
<br><br>
|
||
<script src="https://use.fontawesome.com/41cc35d596.js"></script>
|
||
|
||
<script>
|
||
function clickProcess(element) {
|
||
let columnName =`column${col}_count`;
|
||
console.log(element)
|
||
console.log(element.classList)
|
||
if (myFloID!=undefined) {
|
||
if (element.classList.value == "fa fa-thumbs-o-up")
|
||
{
|
||
let articlekey =articlekeey
|
||
let mainObj = {
|
||
floID: myFloID,
|
||
articlekey: articlekey,
|
||
};
|
||
console.log("mainObj")
|
||
floCloudAPI.sendGeneralData(JSON.stringify(mainObj), "test")
|
||
.then((result, mainObj) => {
|
||
element.classList.value = "fa fa-thumbs-up";
|
||
|
||
if(document.getElementById('num').textContent==" "){
|
||
document.getElementById('num').textContent="1";
|
||
}
|
||
else{
|
||
num1=Number(document.getElementById('num').textContent)
|
||
document.getElementById('num').textContent=num1+1;
|
||
|
||
}
|
||
});
|
||
}
|
||
} else {
|
||
document.getElementById('notify').push("Please login to upvote",{pinned:true});
|
||
}
|
||
}
|
||
|
||
</script>
|
||
|
||
<script id="onLoadStartUp">
|
||
var myFloID,col,articlekeey,myPrivKey;
|
||
function onLoadStartUp(){
|
||
floDapps.launchStartUp().then(result => {
|
||
console.log(result)
|
||
console.log(`Welcome FLO_ID: ${myFloID}`)
|
||
//App functions....
|
||
}).catch(error => console.error(error))
|
||
|
||
var b=JSON.parse(sessionStorage.getItem("z"))
|
||
myFloID=(sessionStorage.getItem("m"))
|
||
col=(sessionStorage.getItem("col"))
|
||
console.log(col)
|
||
articlekeey=(sessionStorage.getItem("articlename"))
|
||
myPrivKey=sessionStorage.getItem("myprivkey");
|
||
myPubKey=sessionStorage.getItem("mypubkey");
|
||
document.getElementById('num').textContent=b;
|
||
//console.log(b+" "+col+" "+myFloID)
|
||
//sessionStorage.removeItem('z');
|
||
}
|
||
</script>
|
||
<script src="./components.min.js"></script>
|
||
<script id="floGlobals">
|
||
/* Constants for FLO blockchain operations !!Make sure to add this at begining!! */
|
||
const floGlobals = {
|
||
//Required for all
|
||
blockchain: "FLO",
|
||
|
||
//Required for blockchain API operators
|
||
apiURL: {
|
||
FLO: ["https://flosight.duckdns.org/", "https://livenet.flocha.in/"],
|
||
FLO_TEST: [
|
||
"https://testnet-flosight.duckdns.org/",
|
||
"https://testnet.flocha.in/",
|
||
],
|
||
},
|
||
adminID: "F9Cs1wF7BS8fDj1gmCsuQ3kpBWRwQj7B2o",
|
||
sendAmt: 0.001,
|
||
fee: 0.0005,
|
||
|
||
//Required for Supernode operations
|
||
SNStorageID: "FEzk75EGMPEQMrCuPosGiwuK162hcEu49E",
|
||
supernodes: {}, //each supnernode must be stored as floID : {uri:<uri>,pubKey:<publicKey>}
|
||
|
||
//for cloud apps
|
||
subAdmins: [],
|
||
webAppURL: "ranchimalltimes-webappserver.duckdns.org",
|
||
application: "TestApp",
|
||
vectorClock: {},
|
||
appObjects: {},
|
||
generalData: {},
|
||
generalVC: {},
|
||
};
|
||
</script>
|
||
|
||
<script id="init_lib">
|
||
//All util libraries required for Standard operations (DO NOT EDIT ANY)
|
||
|
||
/* Reactor Event handling */
|
||
if (typeof reactor == "undefined" || !reactor) {
|
||
(function () {
|
||
function Event(name) {
|
||
this.name = name;
|
||
this.callbacks = [];
|
||
}
|
||
|
||
Event.prototype.registerCallback = function (callback) {
|
||
this.callbacks.push(callback);
|
||
};
|
||
|
||
function Reactor() {
|
||
this.events = {};
|
||
}
|
||
|
||
Reactor.prototype.registerEvent = function (eventName) {
|
||
var event = new Event(eventName);
|
||
this.events[eventName] = event;
|
||
};
|
||
|
||
Reactor.prototype.dispatchEvent = function (eventName, eventArgs) {
|
||
this.events[eventName].callbacks.forEach(function (callback) {
|
||
callback(eventArgs);
|
||
});
|
||
};
|
||
|
||
Reactor.prototype.addEventListener = function (eventName, callback) {
|
||
this.events[eventName].registerCallback(callback);
|
||
};
|
||
|
||
window.reactor = new Reactor();
|
||
})();
|
||
}
|
||
|
||
/* Sample Usage
|
||
--Creating and defining the event--
|
||
reactor.registerEvent('<eventName>');
|
||
reactor.addEventListener('<eventName>', function(someObject){
|
||
do something...
|
||
});
|
||
--Firing the event--
|
||
reactor.dispatchEvent('<eventName>',<someObject>);
|
||
*/
|
||
|
||
/*!
|
||
* Crypto-JS v2.5.4 Crypto.js
|
||
* http://code.google.com/p/crypto-js/
|
||
* Copyright (c) 2009-2013, Jeff Mott. All rights reserved.
|
||
* http://code.google.com/p/crypto-js/wiki/License
|
||
*/
|
||
if (typeof Crypto == "undefined" || !Crypto.util) {
|
||
(function () {
|
||
var base64map =
|
||
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
|
||
|
||
// Global Crypto object
|
||
var Crypto = (window.Crypto = {});
|
||
|
||
// Crypto utilities
|
||
var util = (Crypto.util = {
|
||
// Bit-wise rotate left
|
||
rotl: function (n, b) {
|
||
return (n << b) | (n >>> (32 - b));
|
||
},
|
||
|
||
// Bit-wise rotate right
|
||
rotr: function (n, b) {
|
||
return (n << (32 - b)) | (n >>> b);
|
||
},
|
||
|
||
// Swap big-endian to little-endian and vice versa
|
||
endian: function (n) {
|
||
// If number given, swap endian
|
||
if (n.constructor == Number) {
|
||
return (
|
||
(util.rotl(n, 8) & 0x00ff00ff) | (util.rotl(n, 24) & 0xff00ff00)
|
||
);
|
||
}
|
||
|
||
// Else, assume array and swap all items
|
||
for (var i = 0; i < n.length; i++) n[i] = util.endian(n[i]);
|
||
return n;
|
||
},
|
||
|
||
// Generate an array of any length of random bytes
|
||
randomBytes: function (n) {
|
||
for (var bytes = []; n > 0; n--)
|
||
bytes.push(Math.floor(Math.random() * 256));
|
||
return bytes;
|
||
},
|
||
|
||
// Convert a byte array to big-endian 32-bit words
|
||
bytesToWords: function (bytes) {
|
||
for (var words = [], i = 0, b = 0; i < bytes.length; i++, b += 8)
|
||
words[b >>> 5] |= (bytes[i] & 0xff) << (24 - (b % 32));
|
||
return words;
|
||
},
|
||
|
||
// Convert big-endian 32-bit words to a byte array
|
||
wordsToBytes: function (words) {
|
||
for (var bytes = [], b = 0; b < words.length * 32; b += 8)
|
||
bytes.push((words[b >>> 5] >>> (24 - (b % 32))) & 0xff);
|
||
return bytes;
|
||
},
|
||
|
||
// Convert a byte array to a hex string
|
||
bytesToHex: function (bytes) {
|
||
for (var hex = [], i = 0; i < bytes.length; i++) {
|
||
hex.push((bytes[i] >>> 4).toString(16));
|
||
hex.push((bytes[i] & 0xf).toString(16));
|
||
}
|
||
return hex.join("");
|
||
},
|
||
|
||
// Convert a hex string to a byte array
|
||
hexToBytes: function (hex) {
|
||
for (var bytes = [], c = 0; c < hex.length; c += 2)
|
||
bytes.push(parseInt(hex.substr(c, 2), 16));
|
||
return bytes;
|
||
},
|
||
|
||
// Convert a byte array to a base-64 string
|
||
bytesToBase64: function (bytes) {
|
||
for (var base64 = [], i = 0; i < bytes.length; i += 3) {
|
||
var triplet =
|
||
(bytes[i] << 16) | (bytes[i + 1] << 8) | bytes[i + 2];
|
||
for (var j = 0; j < 4; j++) {
|
||
if (i * 8 + j * 6 <= bytes.length * 8)
|
||
base64.push(
|
||
base64map.charAt((triplet >>> (6 * (3 - j))) & 0x3f)
|
||
);
|
||
else base64.push("=");
|
||
}
|
||
}
|
||
|
||
return base64.join("");
|
||
},
|
||
|
||
// Convert a base-64 string to a byte array
|
||
base64ToBytes: function (base64) {
|
||
// Remove non-base-64 characters
|
||
base64 = base64.replace(/[^A-Z0-9+\/]/gi, "");
|
||
|
||
for (
|
||
var bytes = [], i = 0, imod4 = 0;
|
||
i < base64.length;
|
||
imod4 = ++i % 4
|
||
) {
|
||
if (imod4 == 0) continue;
|
||
bytes.push(
|
||
((base64map.indexOf(base64.charAt(i - 1)) &
|
||
(Math.pow(2, -2 * imod4 + 8) - 1)) <<
|
||
(imod4 * 2)) |
|
||
(base64map.indexOf(base64.charAt(i)) >>> (6 - imod4 * 2))
|
||
);
|
||
}
|
||
|
||
return bytes;
|
||
},
|
||
});
|
||
|
||
// Crypto character encodings
|
||
var charenc = (Crypto.charenc = {});
|
||
|
||
// UTF-8 encoding
|
||
var UTF8 = (charenc.UTF8 = {
|
||
// Convert a string to a byte array
|
||
stringToBytes: function (str) {
|
||
return Binary.stringToBytes(unescape(encodeURIComponent(str)));
|
||
},
|
||
|
||
// Convert a byte array to a string
|
||
bytesToString: function (bytes) {
|
||
return decodeURIComponent(escape(Binary.bytesToString(bytes)));
|
||
},
|
||
});
|
||
|
||
// Binary encoding
|
||
var Binary = (charenc.Binary = {
|
||
// Convert a string to a byte array
|
||
stringToBytes: function (str) {
|
||
for (var bytes = [], i = 0; i < str.length; i++)
|
||
bytes.push(str.charCodeAt(i) & 0xff);
|
||
return bytes;
|
||
},
|
||
|
||
// Convert a byte array to a string
|
||
bytesToString: function (bytes) {
|
||
for (var str = [], i = 0; i < bytes.length; i++)
|
||
str.push(String.fromCharCode(bytes[i]));
|
||
return str.join("");
|
||
},
|
||
});
|
||
})();
|
||
}
|
||
|
||
//Adding SHA1 to fix basic PKBDF2
|
||
/*
|
||
* Crypto-JS v2.5.4
|
||
* http://code.google.com/p/crypto-js/
|
||
* (c) 2009-2012 by Jeff Mott. All rights reserved.
|
||
* http://code.google.com/p/crypto-js/wiki/License
|
||
*/
|
||
(function () {
|
||
// Shortcuts
|
||
var C = Crypto,
|
||
util = C.util,
|
||
charenc = C.charenc,
|
||
UTF8 = charenc.UTF8,
|
||
Binary = charenc.Binary;
|
||
|
||
// Public API
|
||
var SHA1 = (C.SHA1 = function (message, options) {
|
||
var digestbytes = util.wordsToBytes(SHA1._sha1(message));
|
||
return options && options.asBytes
|
||
? digestbytes
|
||
: options && options.asString
|
||
? Binary.bytesToString(digestbytes)
|
||
: util.bytesToHex(digestbytes);
|
||
});
|
||
|
||
// The core
|
||
SHA1._sha1 = function (message) {
|
||
// Convert to byte array
|
||
if (message.constructor == String)
|
||
message = UTF8.stringToBytes(message);
|
||
/* else, assume byte array already */
|
||
|
||
var m = util.bytesToWords(message),
|
||
l = message.length * 8,
|
||
w = [],
|
||
H0 = 1732584193,
|
||
H1 = -271733879,
|
||
H2 = -1732584194,
|
||
H3 = 271733878,
|
||
H4 = -1009589776;
|
||
|
||
// Padding
|
||
m[l >> 5] |= 0x80 << (24 - (l % 32));
|
||
m[(((l + 64) >>> 9) << 4) + 15] = l;
|
||
|
||
for (var i = 0; i < m.length; i += 16) {
|
||
var a = H0,
|
||
b = H1,
|
||
c = H2,
|
||
d = H3,
|
||
e = H4;
|
||
|
||
for (var j = 0; j < 80; j++) {
|
||
if (j < 16) w[j] = m[i + j];
|
||
else {
|
||
var n = w[j - 3] ^ w[j - 8] ^ w[j - 14] ^ w[j - 16];
|
||
w[j] = (n << 1) | (n >>> 31);
|
||
}
|
||
|
||
var t =
|
||
((H0 << 5) | (H0 >>> 27)) +
|
||
H4 +
|
||
(w[j] >>> 0) +
|
||
(j < 20
|
||
? ((H1 & H2) | (~H1 & H3)) + 1518500249
|
||
: j < 40
|
||
? (H1 ^ H2 ^ H3) + 1859775393
|
||
: j < 60
|
||
? ((H1 & H2) | (H1 & H3) | (H2 & H3)) - 1894007588
|
||
: (H1 ^ H2 ^ H3) - 899497514);
|
||
|
||
H4 = H3;
|
||
H3 = H2;
|
||
H2 = (H1 << 30) | (H1 >>> 2);
|
||
H1 = H0;
|
||
H0 = t;
|
||
}
|
||
|
||
H0 += a;
|
||
H1 += b;
|
||
H2 += c;
|
||
H3 += d;
|
||
H4 += e;
|
||
}
|
||
|
||
return [H0, H1, H2, H3, H4];
|
||
};
|
||
|
||
// Package private blocksize
|
||
SHA1._blocksize = 16;
|
||
|
||
SHA1._digestsize = 20;
|
||
})();
|
||
|
||
//Added to make PKBDF2 work
|
||
/*
|
||
* Crypto-JS v2.5.4
|
||
* http://code.google.com/p/crypto-js/
|
||
* (c) 2009-2012 by Jeff Mott. All rights reserved.
|
||
* http://code.google.com/p/crypto-js/wiki/License
|
||
*/
|
||
(function () {
|
||
// Shortcuts
|
||
var C = Crypto,
|
||
util = C.util,
|
||
charenc = C.charenc,
|
||
UTF8 = charenc.UTF8,
|
||
Binary = charenc.Binary;
|
||
|
||
C.HMAC = function (hasher, message, key, options) {
|
||
// Convert to byte arrays
|
||
if (message.constructor == String)
|
||
message = UTF8.stringToBytes(message);
|
||
if (key.constructor == String) key = UTF8.stringToBytes(key);
|
||
/* else, assume byte arrays already */
|
||
|
||
// Allow arbitrary length keys
|
||
if (key.length > hasher._blocksize * 4)
|
||
key = hasher(key, {
|
||
asBytes: true,
|
||
});
|
||
|
||
// XOR keys with pad constants
|
||
var okey = key.slice(0),
|
||
ikey = key.slice(0);
|
||
for (var i = 0; i < hasher._blocksize * 4; i++) {
|
||
okey[i] ^= 0x5c;
|
||
ikey[i] ^= 0x36;
|
||
}
|
||
|
||
var hmacbytes = hasher(
|
||
okey.concat(
|
||
hasher(ikey.concat(message), {
|
||
asBytes: true,
|
||
})
|
||
),
|
||
{
|
||
asBytes: true,
|
||
}
|
||
);
|
||
|
||
return options && options.asBytes
|
||
? hmacbytes
|
||
: options && options.asString
|
||
? Binary.bytesToString(hmacbytes)
|
||
: util.bytesToHex(hmacbytes);
|
||
};
|
||
})();
|
||
|
||
//crypto-sha256-hmac.js
|
||
/*
|
||
* Crypto-JS v2.5.4
|
||
* http://code.google.com/p/crypto-js/
|
||
* (c) 2009-2012 by Jeff Mott. All rights reserved.
|
||
* http://code.google.com/p/crypto-js/wiki/License
|
||
*/
|
||
|
||
function ascii_to_hexa(str) {
|
||
var arr1 = [];
|
||
for (var n = 0, l = str.length; n < l; n++) {
|
||
var hex = Number(str.charCodeAt(n)).toString(16);
|
||
arr1.push(hex);
|
||
}
|
||
return arr1.join("");
|
||
}
|
||
|
||
(typeof Crypto == "undefined" || !Crypto.util) &&
|
||
(function () {
|
||
var d = (window.Crypto = {}),
|
||
k = (d.util = {
|
||
rotl: function (b, a) {
|
||
return (b << a) | (b >>> (32 - a));
|
||
},
|
||
rotr: function (b, a) {
|
||
return (b << (32 - a)) | (b >>> a);
|
||
},
|
||
endian: function (b) {
|
||
if (b.constructor == Number)
|
||
return (k.rotl(b, 8) & 16711935) | (k.rotl(b, 24) & 4278255360);
|
||
for (var a = 0; a < b.length; a++) b[a] = k.endian(b[a]);
|
||
return b;
|
||
},
|
||
randomBytes: function (b) {
|
||
for (var a = []; b > 0; b--)
|
||
a.push(Math.floor(Math.random() * 256));
|
||
return a;
|
||
},
|
||
bytesToWords: function (b) {
|
||
for (var a = [], c = 0, e = 0; c < b.length; c++, e += 8)
|
||
a[e >>> 5] |= (b[c] & 255) << (24 - (e % 32));
|
||
return a;
|
||
},
|
||
wordsToBytes: function (b) {
|
||
for (var a = [], c = 0; c < b.length * 32; c += 8)
|
||
a.push((b[c >>> 5] >>> (24 - (c % 32))) & 255);
|
||
return a;
|
||
},
|
||
bytesToHex: function (b) {
|
||
for (var a = [], c = 0; c < b.length; c++)
|
||
a.push((b[c] >>> 4).toString(16)),
|
||
a.push((b[c] & 15).toString(16));
|
||
return a.join("");
|
||
},
|
||
hexToBytes: function (b) {
|
||
for (var a = [], c = 0; c < b.length; c += 2)
|
||
a.push(parseInt(b.substr(c, 2), 16));
|
||
return a;
|
||
},
|
||
bytesToBase64: function (b) {
|
||
for (var a = [], c = 0; c < b.length; c += 3)
|
||
for (
|
||
var e = (b[c] << 16) | (b[c + 1] << 8) | b[c + 2], p = 0;
|
||
p < 4;
|
||
p++
|
||
)
|
||
c * 8 + p * 6 <= b.length * 8
|
||
? a.push(
|
||
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".charAt(
|
||
(e >>> (6 * (3 - p))) & 63
|
||
)
|
||
)
|
||
: a.push("=");
|
||
return a.join("");
|
||
},
|
||
base64ToBytes: function (b) {
|
||
for (
|
||
var b = b.replace(/[^A-Z0-9+\/]/gi, ""), a = [], c = 0, e = 0;
|
||
c < b.length;
|
||
e = ++c % 4
|
||
)
|
||
e != 0 &&
|
||
a.push(
|
||
(("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".indexOf(
|
||
b.charAt(c - 1)
|
||
) &
|
||
(Math.pow(2, -2 * e + 8) - 1)) <<
|
||
(e * 2)) |
|
||
("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".indexOf(
|
||
b.charAt(c)
|
||
) >>>
|
||
(6 - e * 2))
|
||
);
|
||
return a;
|
||
},
|
||
}),
|
||
d = (d.charenc = {});
|
||
d.UTF8 = {
|
||
stringToBytes: function (b) {
|
||
return g.stringToBytes(unescape(encodeURIComponent(b)));
|
||
},
|
||
bytesToString: function (b) {
|
||
return decodeURIComponent(escape(g.bytesToString(b)));
|
||
},
|
||
};
|
||
var g = (d.Binary = {
|
||
stringToBytes: function (b) {
|
||
for (var a = [], c = 0; c < b.length; c++)
|
||
a.push(b.charCodeAt(c) & 255);
|
||
return a;
|
||
},
|
||
bytesToString: function (b) {
|
||
for (var a = [], c = 0; c < b.length; c++)
|
||
a.push(String.fromCharCode(b[c]));
|
||
return a.join("");
|
||
},
|
||
});
|
||
})();
|
||
(function () {
|
||
var d = Crypto,
|
||
k = d.util,
|
||
g = d.charenc,
|
||
b = g.UTF8,
|
||
a = g.Binary,
|
||
c = [
|
||
1116352408, 1899447441, 3049323471, 3921009573, 961987163, 1508970993,
|
||
2453635748, 2870763221, 3624381080, 310598401, 607225278, 1426881987,
|
||
1925078388, 2162078206, 2614888103, 3248222580, 3835390401,
|
||
4022224774, 264347078, 604807628, 770255983, 1249150122, 1555081692,
|
||
1996064986, 2554220882, 2821834349, 2952996808, 3210313671,
|
||
3336571891, 3584528711, 113926993, 338241895, 666307205, 773529912,
|
||
1294757372, 1396182291, 1695183700, 1986661051, 2177026350,
|
||
2456956037, 2730485921, 2820302411, 3259730800, 3345764771,
|
||
3516065817, 3600352804, 4094571909, 275423344, 430227734, 506948616,
|
||
659060556, 883997877, 958139571, 1322822218, 1537002063, 1747873779,
|
||
1955562222, 2024104815, 2227730452, 2361852424, 2428436474,
|
||
2756734187, 3204031479, 3329325298,
|
||
],
|
||
e = (d.SHA256 = function (b, c) {
|
||
var f = k.wordsToBytes(e._sha256(b));
|
||
return c && c.asBytes
|
||
? f
|
||
: c && c.asString
|
||
? a.bytesToString(f)
|
||
: k.bytesToHex(f);
|
||
});
|
||
e._sha256 = function (a) {
|
||
a.constructor == String && (a = b.stringToBytes(a));
|
||
var e = k.bytesToWords(a),
|
||
f = a.length * 8,
|
||
a = [
|
||
1779033703, 3144134277, 1013904242, 2773480762, 1359893119,
|
||
2600822924, 528734635, 1541459225,
|
||
],
|
||
d = [],
|
||
g,
|
||
m,
|
||
r,
|
||
i,
|
||
n,
|
||
o,
|
||
s,
|
||
t,
|
||
h,
|
||
l,
|
||
j;
|
||
e[f >> 5] |= 128 << (24 - (f % 32));
|
||
e[(((f + 64) >> 9) << 4) + 15] = f;
|
||
for (t = 0; t < e.length; t += 16) {
|
||
f = a[0];
|
||
g = a[1];
|
||
m = a[2];
|
||
r = a[3];
|
||
i = a[4];
|
||
n = a[5];
|
||
o = a[6];
|
||
s = a[7];
|
||
for (h = 0; h < 64; h++) {
|
||
h < 16
|
||
? (d[h] = e[h + t])
|
||
: ((l = d[h - 15]),
|
||
(j = d[h - 2]),
|
||
(d[h] =
|
||
(((l << 25) | (l >>> 7)) ^
|
||
((l << 14) | (l >>> 18)) ^
|
||
(l >>> 3)) +
|
||
(d[h - 7] >>> 0) +
|
||
(((j << 15) | (j >>> 17)) ^
|
||
((j << 13) | (j >>> 19)) ^
|
||
(j >>> 10)) +
|
||
(d[h - 16] >>> 0)));
|
||
j = (f & g) ^ (f & m) ^ (g & m);
|
||
var u =
|
||
((f << 30) | (f >>> 2)) ^
|
||
((f << 19) | (f >>> 13)) ^
|
||
((f << 10) | (f >>> 22));
|
||
l =
|
||
(s >>> 0) +
|
||
(((i << 26) | (i >>> 6)) ^
|
||
((i << 21) | (i >>> 11)) ^
|
||
((i << 7) | (i >>> 25))) +
|
||
((i & n) ^ (~i & o)) +
|
||
c[h] +
|
||
(d[h] >>> 0);
|
||
j = u + j;
|
||
s = o;
|
||
o = n;
|
||
n = i;
|
||
i = (r + l) >>> 0;
|
||
r = m;
|
||
m = g;
|
||
g = f;
|
||
f = (l + j) >>> 0;
|
||
}
|
||
a[0] += f;
|
||
a[1] += g;
|
||
a[2] += m;
|
||
a[3] += r;
|
||
a[4] += i;
|
||
a[5] += n;
|
||
a[6] += o;
|
||
a[7] += s;
|
||
}
|
||
return a;
|
||
};
|
||
e._blocksize = 16;
|
||
e._digestsize = 32;
|
||
})();
|
||
(function () {
|
||
var d = Crypto,
|
||
k = d.util,
|
||
g = d.charenc,
|
||
b = g.UTF8,
|
||
a = g.Binary;
|
||
d.HMAC = function (c, e, d, g) {
|
||
e.constructor == String && (e = b.stringToBytes(e));
|
||
d.constructor == String && (d = b.stringToBytes(d));
|
||
d.length > c._blocksize * 4 &&
|
||
(d = c(d, {
|
||
asBytes: !0,
|
||
}));
|
||
for (
|
||
var f = d.slice(0), d = d.slice(0), q = 0;
|
||
q < c._blocksize * 4;
|
||
q++
|
||
)
|
||
(f[q] ^= 92), (d[q] ^= 54);
|
||
c = c(
|
||
f.concat(
|
||
c(d.concat(e), {
|
||
asBytes: !0,
|
||
})
|
||
),
|
||
{
|
||
asBytes: !0,
|
||
}
|
||
);
|
||
return g && g.asBytes
|
||
? c
|
||
: g && g.asString
|
||
? a.bytesToString(c)
|
||
: k.bytesToHex(c);
|
||
};
|
||
})();
|
||
|
||
/*!
|
||
* Random number generator with ArcFour PRNG
|
||
*
|
||
* NOTE: For best results, put code like
|
||
* <body onclick='SecureRandom.seedTime();' onkeypress='SecureRandom.seedTime();'>
|
||
* in your main HTML document.
|
||
*
|
||
* Copyright Tom Wu, bitaddress.org BSD License.
|
||
* http://www-cs-students.stanford.edu/~tjw/jsbn/LICENSE
|
||
*/
|
||
(function () {
|
||
// Constructor function of Global SecureRandom object
|
||
var sr = (window.SecureRandom = function () {});
|
||
|
||
// Properties
|
||
sr.state;
|
||
sr.pool;
|
||
sr.pptr;
|
||
sr.poolCopyOnInit;
|
||
|
||
// Pool size must be a multiple of 4 and greater than 32.
|
||
// An array of bytes the size of the pool will be passed to init()
|
||
sr.poolSize = 256;
|
||
|
||
// --- object methods ---
|
||
|
||
// public method
|
||
// ba: byte array
|
||
sr.prototype.nextBytes = function (ba) {
|
||
var i;
|
||
if (
|
||
window.crypto &&
|
||
window.crypto.getRandomValues &&
|
||
window.Uint8Array
|
||
) {
|
||
try {
|
||
var rvBytes = new Uint8Array(ba.length);
|
||
window.crypto.getRandomValues(rvBytes);
|
||
for (i = 0; i < ba.length; ++i) ba[i] = sr.getByte() ^ rvBytes[i];
|
||
return;
|
||
} catch (e) {
|
||
alert(e);
|
||
}
|
||
}
|
||
for (i = 0; i < ba.length; ++i) ba[i] = sr.getByte();
|
||
};
|
||
|
||
// --- static methods ---
|
||
|
||
// Mix in the current time (w/milliseconds) into the pool
|
||
// NOTE: this method should be called from body click/keypress event handlers to increase entropy
|
||
sr.seedTime = function () {
|
||
sr.seedInt(new Date().getTime());
|
||
};
|
||
|
||
sr.getByte = function () {
|
||
if (sr.state == null) {
|
||
sr.seedTime();
|
||
sr.state = sr.ArcFour(); // Plug in your RNG constructor here
|
||
sr.state.init(sr.pool);
|
||
sr.poolCopyOnInit = [];
|
||
for (sr.pptr = 0; sr.pptr < sr.pool.length; ++sr.pptr)
|
||
sr.poolCopyOnInit[sr.pptr] = sr.pool[sr.pptr];
|
||
sr.pptr = 0;
|
||
}
|
||
// TODO: allow reseeding after first request
|
||
return sr.state.next();
|
||
};
|
||
|
||
// Mix in a 32-bit integer into the pool
|
||
sr.seedInt = function (x) {
|
||
sr.seedInt8(x);
|
||
sr.seedInt8(x >> 8);
|
||
sr.seedInt8(x >> 16);
|
||
sr.seedInt8(x >> 24);
|
||
};
|
||
|
||
// Mix in a 16-bit integer into the pool
|
||
sr.seedInt16 = function (x) {
|
||
sr.seedInt8(x);
|
||
sr.seedInt8(x >> 8);
|
||
};
|
||
|
||
// Mix in a 8-bit integer into the pool
|
||
sr.seedInt8 = function (x) {
|
||
sr.pool[sr.pptr++] ^= x & 255;
|
||
if (sr.pptr >= sr.poolSize) sr.pptr -= sr.poolSize;
|
||
};
|
||
|
||
// Arcfour is a PRNG
|
||
sr.ArcFour = function () {
|
||
function Arcfour() {
|
||
this.i = 0;
|
||
this.j = 0;
|
||
this.S = new Array();
|
||
}
|
||
|
||
// Initialize arcfour context from key, an array of ints, each from [0..255]
|
||
function ARC4init(key) {
|
||
var i, j, t;
|
||
for (i = 0; i < 256; ++i) this.S[i] = i;
|
||
j = 0;
|
||
for (i = 0; i < 256; ++i) {
|
||
j = (j + this.S[i] + key[i % key.length]) & 255;
|
||
t = this.S[i];
|
||
this.S[i] = this.S[j];
|
||
this.S[j] = t;
|
||
}
|
||
this.i = 0;
|
||
this.j = 0;
|
||
}
|
||
|
||
function ARC4next() {
|
||
var t;
|
||
this.i = (this.i + 1) & 255;
|
||
this.j = (this.j + this.S[this.i]) & 255;
|
||
t = this.S[this.i];
|
||
this.S[this.i] = this.S[this.j];
|
||
this.S[this.j] = t;
|
||
return this.S[(t + this.S[this.i]) & 255];
|
||
}
|
||
|
||
Arcfour.prototype.init = ARC4init;
|
||
Arcfour.prototype.next = ARC4next;
|
||
|
||
return new Arcfour();
|
||
};
|
||
|
||
// Initialize the pool with junk if needed.
|
||
if (sr.pool == null) {
|
||
sr.pool = new Array();
|
||
sr.pptr = 0;
|
||
var t;
|
||
if (
|
||
window.crypto &&
|
||
window.crypto.getRandomValues &&
|
||
window.Uint8Array
|
||
) {
|
||
try {
|
||
// Use webcrypto if available
|
||
var ua = new Uint8Array(sr.poolSize);
|
||
window.crypto.getRandomValues(ua);
|
||
for (t = 0; t < sr.poolSize; ++t) sr.pool[sr.pptr++] = ua[t];
|
||
} catch (e) {
|
||
alert(e);
|
||
}
|
||
}
|
||
while (sr.pptr < sr.poolSize) {
|
||
// extract some randomness from Math.random()
|
||
t = Math.floor(65536 * Math.random());
|
||
sr.pool[sr.pptr++] = t >>> 8;
|
||
sr.pool[sr.pptr++] = t & 255;
|
||
}
|
||
sr.pptr = Math.floor(sr.poolSize * Math.random());
|
||
sr.seedTime();
|
||
// entropy
|
||
var entropyStr = "";
|
||
// screen size and color depth: ~4.8 to ~5.4 bits
|
||
entropyStr +=
|
||
window.screen.height * window.screen.width * window.screen.colorDepth;
|
||
entropyStr +=
|
||
window.screen.availHeight *
|
||
window.screen.availWidth *
|
||
window.screen.pixelDepth;
|
||
// time zone offset: ~4 bits
|
||
var dateObj = new Date();
|
||
var timeZoneOffset = dateObj.getTimezoneOffset();
|
||
entropyStr += timeZoneOffset;
|
||
// user agent: ~8.3 to ~11.6 bits
|
||
entropyStr += navigator.userAgent;
|
||
// browser plugin details: ~16.2 to ~21.8 bits
|
||
var pluginsStr = "";
|
||
for (var i = 0; i < navigator.plugins.length; i++) {
|
||
pluginsStr +=
|
||
navigator.plugins[i].name +
|
||
" " +
|
||
navigator.plugins[i].filename +
|
||
" " +
|
||
navigator.plugins[i].description +
|
||
" " +
|
||
navigator.plugins[i].version +
|
||
", ";
|
||
}
|
||
var mimeTypesStr = "";
|
||
for (var i = 0; i < navigator.mimeTypes.length; i++) {
|
||
mimeTypesStr +=
|
||
navigator.mimeTypes[i].description +
|
||
" " +
|
||
navigator.mimeTypes[i].type +
|
||
" " +
|
||
navigator.mimeTypes[i].suffixes +
|
||
", ";
|
||
}
|
||
entropyStr += pluginsStr + mimeTypesStr;
|
||
// cookies and storage: 1 bit
|
||
entropyStr +=
|
||
navigator.cookieEnabled + typeof sessionStorage + typeof localStorage;
|
||
// language: ~7 bit
|
||
entropyStr += navigator.language;
|
||
// history: ~2 bit
|
||
entropyStr += window.history.length;
|
||
// location
|
||
entropyStr += window.location;
|
||
|
||
var entropyBytes = Crypto.SHA256(entropyStr, {
|
||
asBytes: true,
|
||
});
|
||
for (var i = 0; i < entropyBytes.length; i++) {
|
||
sr.seedInt8(entropyBytes[i]);
|
||
}
|
||
}
|
||
})();
|
||
|
||
//ripemd160.js
|
||
/*
|
||
CryptoJS v3.1.2
|
||
code.google.com/p/crypto-js
|
||
(c) 2009-2013 by Jeff Mott. All rights reserved.
|
||
code.google.com/p/crypto-js/wiki/License
|
||
*/
|
||
/** @preserve
|
||
(c) 2012 by Cédric Mesnil. All rights reserved.
|
||
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
|
||
- Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
|
||
- Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
|
||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
*/
|
||
|
||
// Constants table
|
||
var zl = [
|
||
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 7, 4, 13, 1, 10, 6,
|
||
15, 3, 12, 0, 9, 5, 2, 14, 11, 8, 3, 10, 14, 4, 9, 15, 8, 1, 2, 7, 0, 6,
|
||
13, 11, 5, 12, 1, 9, 11, 10, 0, 8, 12, 4, 13, 3, 7, 15, 14, 5, 6, 2, 4, 0,
|
||
5, 9, 7, 12, 2, 10, 14, 1, 3, 8, 11, 6, 15, 13,
|
||
];
|
||
var zr = [
|
||
5, 14, 7, 0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12, 6, 11, 3, 7, 0, 13,
|
||
5, 10, 14, 15, 8, 12, 4, 9, 1, 2, 15, 5, 1, 3, 7, 14, 6, 9, 11, 8, 12, 2,
|
||
10, 0, 4, 13, 8, 6, 4, 1, 3, 11, 15, 0, 5, 12, 2, 13, 9, 7, 10, 14, 12,
|
||
15, 10, 4, 1, 5, 8, 7, 6, 2, 13, 14, 0, 3, 9, 11,
|
||
];
|
||
var sl = [
|
||
11, 14, 15, 12, 5, 8, 7, 9, 11, 13, 14, 15, 6, 7, 9, 8, 7, 6, 8, 13, 11,
|
||
9, 7, 15, 7, 12, 15, 9, 11, 7, 13, 12, 11, 13, 6, 7, 14, 9, 13, 15, 14, 8,
|
||
13, 6, 5, 12, 7, 5, 11, 12, 14, 15, 14, 15, 9, 8, 9, 14, 5, 6, 8, 6, 5,
|
||
12, 9, 15, 5, 11, 6, 8, 13, 12, 5, 12, 13, 14, 11, 8, 5, 6,
|
||
];
|
||
var sr = [
|
||
8, 9, 9, 11, 13, 15, 15, 5, 7, 7, 8, 11, 14, 14, 12, 6, 9, 13, 15, 7, 12,
|
||
8, 9, 11, 7, 7, 12, 7, 6, 15, 13, 11, 9, 7, 15, 11, 8, 6, 6, 14, 12, 13,
|
||
5, 14, 13, 13, 7, 5, 15, 5, 8, 11, 14, 14, 6, 14, 6, 9, 12, 9, 12, 5, 15,
|
||
8, 8, 5, 12, 9, 12, 5, 14, 6, 8, 13, 6, 5, 15, 13, 11, 11,
|
||
];
|
||
|
||
var hl = [0x00000000, 0x5a827999, 0x6ed9eba1, 0x8f1bbcdc, 0xa953fd4e];
|
||
var hr = [0x50a28be6, 0x5c4dd124, 0x6d703ef3, 0x7a6d76e9, 0x00000000];
|
||
|
||
var bytesToWords = function (bytes) {
|
||
var words = [];
|
||
for (var i = 0, b = 0; i < bytes.length; i++, b += 8) {
|
||
words[b >>> 5] |= bytes[i] << (24 - (b % 32));
|
||
}
|
||
return words;
|
||
};
|
||
|
||
var wordsToBytes = function (words) {
|
||
var bytes = [];
|
||
for (var b = 0; b < words.length * 32; b += 8) {
|
||
bytes.push((words[b >>> 5] >>> (24 - (b % 32))) & 0xff);
|
||
}
|
||
return bytes;
|
||
};
|
||
|
||
var processBlock = function (H, M, offset) {
|
||
// Swap endian
|
||
for (var i = 0; i < 16; i++) {
|
||
var offset_i = offset + i;
|
||
var M_offset_i = M[offset_i];
|
||
|
||
// Swap
|
||
M[offset_i] =
|
||
(((M_offset_i << 8) | (M_offset_i >>> 24)) & 0x00ff00ff) |
|
||
(((M_offset_i << 24) | (M_offset_i >>> 8)) & 0xff00ff00);
|
||
}
|
||
|
||
// Working variables
|
||
var al, bl, cl, dl, el;
|
||
var ar, br, cr, dr, er;
|
||
|
||
ar = al = H[0];
|
||
br = bl = H[1];
|
||
cr = cl = H[2];
|
||
dr = dl = H[3];
|
||
er = el = H[4];
|
||
// Computation
|
||
var t;
|
||
for (var i = 0; i < 80; i += 1) {
|
||
t = (al + M[offset + zl[i]]) | 0;
|
||
if (i < 16) {
|
||
t += f1(bl, cl, dl) + hl[0];
|
||
} else if (i < 32) {
|
||
t += f2(bl, cl, dl) + hl[1];
|
||
} else if (i < 48) {
|
||
t += f3(bl, cl, dl) + hl[2];
|
||
} else if (i < 64) {
|
||
t += f4(bl, cl, dl) + hl[3];
|
||
} else {
|
||
// if (i<80) {
|
||
t += f5(bl, cl, dl) + hl[4];
|
||
}
|
||
t = t | 0;
|
||
t = rotl(t, sl[i]);
|
||
t = (t + el) | 0;
|
||
al = el;
|
||
el = dl;
|
||
dl = rotl(cl, 10);
|
||
cl = bl;
|
||
bl = t;
|
||
|
||
t = (ar + M[offset + zr[i]]) | 0;
|
||
if (i < 16) {
|
||
t += f5(br, cr, dr) + hr[0];
|
||
} else if (i < 32) {
|
||
t += f4(br, cr, dr) + hr[1];
|
||
} else if (i < 48) {
|
||
t += f3(br, cr, dr) + hr[2];
|
||
} else if (i < 64) {
|
||
t += f2(br, cr, dr) + hr[3];
|
||
} else {
|
||
// if (i<80) {
|
||
t += f1(br, cr, dr) + hr[4];
|
||
}
|
||
t = t | 0;
|
||
t = rotl(t, sr[i]);
|
||
t = (t + er) | 0;
|
||
ar = er;
|
||
er = dr;
|
||
dr = rotl(cr, 10);
|
||
cr = br;
|
||
br = t;
|
||
}
|
||
// Intermediate hash value
|
||
t = (H[1] + cl + dr) | 0;
|
||
H[1] = (H[2] + dl + er) | 0;
|
||
H[2] = (H[3] + el + ar) | 0;
|
||
H[3] = (H[4] + al + br) | 0;
|
||
H[4] = (H[0] + bl + cr) | 0;
|
||
H[0] = t;
|
||
};
|
||
|
||
function f1(x, y, z) {
|
||
return x ^ y ^ z;
|
||
}
|
||
|
||
function f2(x, y, z) {
|
||
return (x & y) | (~x & z);
|
||
}
|
||
|
||
function f3(x, y, z) {
|
||
return (x | ~y) ^ z;
|
||
}
|
||
|
||
function f4(x, y, z) {
|
||
return (x & z) | (y & ~z);
|
||
}
|
||
|
||
function f5(x, y, z) {
|
||
return x ^ (y | ~z);
|
||
}
|
||
|
||
function rotl(x, n) {
|
||
return (x << n) | (x >>> (32 - n));
|
||
}
|
||
|
||
function ripemd160(message) {
|
||
var H = [0x67452301, 0xefcdab89, 0x98badcfe, 0x10325476, 0xc3d2e1f0];
|
||
|
||
var m = bytesToWords(message);
|
||
|
||
var nBitsLeft = message.length * 8;
|
||
var nBitsTotal = message.length * 8;
|
||
|
||
// Add padding
|
||
m[nBitsLeft >>> 5] |= 0x80 << (24 - (nBitsLeft % 32));
|
||
m[(((nBitsLeft + 64) >>> 9) << 4) + 14] =
|
||
(((nBitsTotal << 8) | (nBitsTotal >>> 24)) & 0x00ff00ff) |
|
||
(((nBitsTotal << 24) | (nBitsTotal >>> 8)) & 0xff00ff00);
|
||
|
||
for (var i = 0; i < m.length; i += 16) {
|
||
processBlock(H, m, i);
|
||
}
|
||
|
||
// Swap endian
|
||
for (var i = 0; i < 5; i++) {
|
||
// Shortcut
|
||
var H_i = H[i];
|
||
|
||
// Swap
|
||
H[i] =
|
||
(((H_i << 8) | (H_i >>> 24)) & 0x00ff00ff) |
|
||
(((H_i << 24) | (H_i >>> 8)) & 0xff00ff00);
|
||
}
|
||
|
||
var digestbytes = wordsToBytes(H);
|
||
return digestbytes;
|
||
}
|
||
|
||
// Upstream 'BigInteger' here:
|
||
// Original Author: http://www-cs-students.stanford.edu/~tjw/jsbn/
|
||
// Follows 'jsbn' on Github: https://github.com/jasondavies/jsbn
|
||
// Review and Testing: https://github.com/cryptocoinjs/bigi/
|
||
/*!
|
||
* Basic JavaScript BN library - subset useful for RSA encryption. v1.4
|
||
*
|
||
* Copyright (c) 2005 Tom Wu
|
||
* All Rights Reserved.
|
||
* BSD License
|
||
* http://www-cs-students.stanford.edu/~tjw/jsbn/LICENSE
|
||
*
|
||
* Copyright Stephan Thomas
|
||
* Copyright pointbiz
|
||
*/
|
||
|
||
(function () {
|
||
// (public) Constructor function of Global BigInteger object
|
||
var BigInteger = (window.BigInteger = function BigInteger(a, b, c) {
|
||
if (!(this instanceof BigInteger)) return new BigInteger(a, b, c);
|
||
|
||
if (a != null)
|
||
if ("number" == typeof a) this.fromNumber(a, b, c);
|
||
else if (b == null && "string" != typeof a) this.fromString(a, 256);
|
||
else this.fromString(a, b);
|
||
});
|
||
|
||
// Bits per digit
|
||
var dbits;
|
||
|
||
// JavaScript engine analysis
|
||
var canary = 0xdeadbeefcafe;
|
||
var j_lm = (canary & 0xffffff) == 0xefcafe;
|
||
|
||
// return new, unset BigInteger
|
||
function nbi() {
|
||
return new BigInteger(null);
|
||
}
|
||
|
||
// am: Compute w_j += (x*this_i), propagate carries,
|
||
// c is initial carry, returns final carry.
|
||
// c < 3*dvalue, x < 2*dvalue, this_i < dvalue
|
||
// We need to select the fastest one that works in this environment.
|
||
|
||
// am1: use a single mult and divide to get the high bits,
|
||
// max digit bits should be 26 because
|
||
// max internal value = 2*dvalue^2-2*dvalue (< 2^53)
|
||
function am1(i, x, w, j, c, n) {
|
||
while (--n >= 0) {
|
||
var v = x * this[i++] + w[j] + c;
|
||
c = Math.floor(v / 0x4000000);
|
||
w[j++] = v & 0x3ffffff;
|
||
}
|
||
return c;
|
||
}
|
||
// am2 avoids a big mult-and-extract completely.
|
||
// Max digit bits should be <= 30 because we do bitwise ops
|
||
// on values up to 2*hdvalue^2-hdvalue-1 (< 2^31)
|
||
function am2(i, x, w, j, c, n) {
|
||
var xl = x & 0x7fff,
|
||
xh = x >> 15;
|
||
while (--n >= 0) {
|
||
var l = this[i] & 0x7fff;
|
||
var h = this[i++] >> 15;
|
||
var m = xh * l + h * xl;
|
||
l = xl * l + ((m & 0x7fff) << 15) + w[j] + (c & 0x3fffffff);
|
||
c = (l >>> 30) + (m >>> 15) + xh * h + (c >>> 30);
|
||
w[j++] = l & 0x3fffffff;
|
||
}
|
||
return c;
|
||
}
|
||
// Alternately, set max digit bits to 28 since some
|
||
// browsers slow down when dealing with 32-bit numbers.
|
||
function am3(i, x, w, j, c, n) {
|
||
var xl = x & 0x3fff,
|
||
xh = x >> 14;
|
||
while (--n >= 0) {
|
||
var l = this[i] & 0x3fff;
|
||
var h = this[i++] >> 14;
|
||
var m = xh * l + h * xl;
|
||
l = xl * l + ((m & 0x3fff) << 14) + w[j] + c;
|
||
c = (l >> 28) + (m >> 14) + xh * h;
|
||
w[j++] = l & 0xfffffff;
|
||
}
|
||
return c;
|
||
}
|
||
if (j_lm && navigator.appName == "Microsoft Internet Explorer") {
|
||
BigInteger.prototype.am = am2;
|
||
dbits = 30;
|
||
} else if (j_lm && navigator.appName != "Netscape") {
|
||
BigInteger.prototype.am = am1;
|
||
dbits = 26;
|
||
} else {
|
||
// Mozilla/Netscape seems to prefer am3
|
||
BigInteger.prototype.am = am3;
|
||
dbits = 28;
|
||
}
|
||
|
||
BigInteger.prototype.DB = dbits;
|
||
BigInteger.prototype.DM = (1 << dbits) - 1;
|
||
BigInteger.prototype.DV = 1 << dbits;
|
||
|
||
var BI_FP = 52;
|
||
BigInteger.prototype.FV = Math.pow(2, BI_FP);
|
||
BigInteger.prototype.F1 = BI_FP - dbits;
|
||
BigInteger.prototype.F2 = 2 * dbits - BI_FP;
|
||
|
||
// Digit conversions
|
||
var BI_RM = "0123456789abcdefghijklmnopqrstuvwxyz";
|
||
var BI_RC = new Array();
|
||
var rr, vv;
|
||
rr = "0".charCodeAt(0);
|
||
for (vv = 0; vv <= 9; ++vv) BI_RC[rr++] = vv;
|
||
rr = "a".charCodeAt(0);
|
||
for (vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;
|
||
rr = "A".charCodeAt(0);
|
||
for (vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;
|
||
|
||
function int2char(n) {
|
||
return BI_RM.charAt(n);
|
||
}
|
||
|
||
function intAt(s, i) {
|
||
var c = BI_RC[s.charCodeAt(i)];
|
||
return c == null ? -1 : c;
|
||
}
|
||
|
||
// return bigint initialized to value
|
||
function nbv(i) {
|
||
var r = nbi();
|
||
r.fromInt(i);
|
||
return r;
|
||
}
|
||
|
||
// returns bit length of the integer x
|
||
function nbits(x) {
|
||
var r = 1,
|
||
t;
|
||
if ((t = x >>> 16) != 0) {
|
||
x = t;
|
||
r += 16;
|
||
}
|
||
if ((t = x >> 8) != 0) {
|
||
x = t;
|
||
r += 8;
|
||
}
|
||
if ((t = x >> 4) != 0) {
|
||
x = t;
|
||
r += 4;
|
||
}
|
||
if ((t = x >> 2) != 0) {
|
||
x = t;
|
||
r += 2;
|
||
}
|
||
if ((t = x >> 1) != 0) {
|
||
x = t;
|
||
r += 1;
|
||
}
|
||
return r;
|
||
}
|
||
|
||
// (protected) copy this to r
|
||
BigInteger.prototype.copyTo = function (r) {
|
||
for (var i = this.t - 1; i >= 0; --i) r[i] = this[i];
|
||
r.t = this.t;
|
||
r.s = this.s;
|
||
};
|
||
|
||
// (protected) set from integer value x, -DV <= x < DV
|
||
BigInteger.prototype.fromInt = function (x) {
|
||
this.t = 1;
|
||
this.s = x < 0 ? -1 : 0;
|
||
if (x > 0) this[0] = x;
|
||
else if (x < -1) this[0] = x + this.DV;
|
||
else this.t = 0;
|
||
};
|
||
|
||
// (protected) set from string and radix
|
||
BigInteger.prototype.fromString = function (s, b) {
|
||
var k;
|
||
if (b == 16) k = 4;
|
||
else if (b == 8) k = 3;
|
||
else if (b == 256) k = 8;
|
||
// byte array
|
||
else if (b == 2) k = 1;
|
||
else if (b == 32) k = 5;
|
||
else if (b == 4) k = 2;
|
||
else {
|
||
this.fromRadix(s, b);
|
||
return;
|
||
}
|
||
this.t = 0;
|
||
this.s = 0;
|
||
var i = s.length,
|
||
mi = false,
|
||
sh = 0;
|
||
while (--i >= 0) {
|
||
var x = k == 8 ? s[i] & 0xff : intAt(s, i);
|
||
if (x < 0) {
|
||
if (s.charAt(i) == "-") mi = true;
|
||
continue;
|
||
}
|
||
mi = false;
|
||
if (sh == 0) this[this.t++] = x;
|
||
else if (sh + k > this.DB) {
|
||
this[this.t - 1] |= (x & ((1 << (this.DB - sh)) - 1)) << sh;
|
||
this[this.t++] = x >> (this.DB - sh);
|
||
} else this[this.t - 1] |= x << sh;
|
||
sh += k;
|
||
if (sh >= this.DB) sh -= this.DB;
|
||
}
|
||
if (k == 8 && (s[0] & 0x80) != 0) {
|
||
this.s = -1;
|
||
if (sh > 0) this[this.t - 1] |= ((1 << (this.DB - sh)) - 1) << sh;
|
||
}
|
||
this.clamp();
|
||
if (mi) BigInteger.ZERO.subTo(this, this);
|
||
};
|
||
|
||
// (protected) clamp off excess high words
|
||
BigInteger.prototype.clamp = function () {
|
||
var c = this.s & this.DM;
|
||
while (this.t > 0 && this[this.t - 1] == c) --this.t;
|
||
};
|
||
|
||
// (protected) r = this << n*DB
|
||
BigInteger.prototype.dlShiftTo = function (n, r) {
|
||
var i;
|
||
for (i = this.t - 1; i >= 0; --i) r[i + n] = this[i];
|
||
for (i = n - 1; i >= 0; --i) r[i] = 0;
|
||
r.t = this.t + n;
|
||
r.s = this.s;
|
||
};
|
||
|
||
// (protected) r = this >> n*DB
|
||
BigInteger.prototype.drShiftTo = function (n, r) {
|
||
for (var i = n; i < this.t; ++i) r[i - n] = this[i];
|
||
r.t = Math.max(this.t - n, 0);
|
||
r.s = this.s;
|
||
};
|
||
|
||
// (protected) r = this << n
|
||
BigInteger.prototype.lShiftTo = function (n, r) {
|
||
var bs = n % this.DB;
|
||
var cbs = this.DB - bs;
|
||
var bm = (1 << cbs) - 1;
|
||
var ds = Math.floor(n / this.DB),
|
||
c = (this.s << bs) & this.DM,
|
||
i;
|
||
for (i = this.t - 1; i >= 0; --i) {
|
||
r[i + ds + 1] = (this[i] >> cbs) | c;
|
||
c = (this[i] & bm) << bs;
|
||
}
|
||
for (i = ds - 1; i >= 0; --i) r[i] = 0;
|
||
r[ds] = c;
|
||
r.t = this.t + ds + 1;
|
||
r.s = this.s;
|
||
r.clamp();
|
||
};
|
||
|
||
// (protected) r = this >> n
|
||
BigInteger.prototype.rShiftTo = function (n, r) {
|
||
r.s = this.s;
|
||
var ds = Math.floor(n / this.DB);
|
||
if (ds >= this.t) {
|
||
r.t = 0;
|
||
return;
|
||
}
|
||
var bs = n % this.DB;
|
||
var cbs = this.DB - bs;
|
||
var bm = (1 << bs) - 1;
|
||
r[0] = this[ds] >> bs;
|
||
for (var i = ds + 1; i < this.t; ++i) {
|
||
r[i - ds - 1] |= (this[i] & bm) << cbs;
|
||
r[i - ds] = this[i] >> bs;
|
||
}
|
||
if (bs > 0) r[this.t - ds - 1] |= (this.s & bm) << cbs;
|
||
r.t = this.t - ds;
|
||
r.clamp();
|
||
};
|
||
|
||
// (protected) r = this - a
|
||
BigInteger.prototype.subTo = function (a, r) {
|
||
var i = 0,
|
||
c = 0,
|
||
m = Math.min(a.t, this.t);
|
||
while (i < m) {
|
||
c += this[i] - a[i];
|
||
r[i++] = c & this.DM;
|
||
c >>= this.DB;
|
||
}
|
||
if (a.t < this.t) {
|
||
c -= a.s;
|
||
while (i < this.t) {
|
||
c += this[i];
|
||
r[i++] = c & this.DM;
|
||
c >>= this.DB;
|
||
}
|
||
c += this.s;
|
||
} else {
|
||
c += this.s;
|
||
while (i < a.t) {
|
||
c -= a[i];
|
||
r[i++] = c & this.DM;
|
||
c >>= this.DB;
|
||
}
|
||
c -= a.s;
|
||
}
|
||
r.s = c < 0 ? -1 : 0;
|
||
if (c < -1) r[i++] = this.DV + c;
|
||
else if (c > 0) r[i++] = c;
|
||
r.t = i;
|
||
r.clamp();
|
||
};
|
||
|
||
// (protected) r = this * a, r != this,a (HAC 14.12)
|
||
// "this" should be the larger one if appropriate.
|
||
BigInteger.prototype.multiplyTo = function (a, r) {
|
||
var x = this.abs(),
|
||
y = a.abs();
|
||
var i = x.t;
|
||
r.t = i + y.t;
|
||
while (--i >= 0) r[i] = 0;
|
||
for (i = 0; i < y.t; ++i) r[i + x.t] = x.am(0, y[i], r, i, 0, x.t);
|
||
r.s = 0;
|
||
r.clamp();
|
||
if (this.s != a.s) BigInteger.ZERO.subTo(r, r);
|
||
};
|
||
|
||
// (protected) r = this^2, r != this (HAC 14.16)
|
||
BigInteger.prototype.squareTo = function (r) {
|
||
var x = this.abs();
|
||
var i = (r.t = 2 * x.t);
|
||
while (--i >= 0) r[i] = 0;
|
||
for (i = 0; i < x.t - 1; ++i) {
|
||
var c = x.am(i, x[i], r, 2 * i, 0, 1);
|
||
if (
|
||
(r[i + x.t] += x.am(
|
||
i + 1,
|
||
2 * x[i],
|
||
r,
|
||
2 * i + 1,
|
||
c,
|
||
x.t - i - 1
|
||
)) >= x.DV
|
||
) {
|
||
r[i + x.t] -= x.DV;
|
||
r[i + x.t + 1] = 1;
|
||
}
|
||
}
|
||
if (r.t > 0) r[r.t - 1] += x.am(i, x[i], r, 2 * i, 0, 1);
|
||
r.s = 0;
|
||
r.clamp();
|
||
};
|
||
|
||
// (protected) divide this by m, quotient and remainder to q, r (HAC 14.20)
|
||
// r != q, this != m. q or r may be null.
|
||
BigInteger.prototype.divRemTo = function (m, q, r) {
|
||
var pm = m.abs();
|
||
if (pm.t <= 0) return;
|
||
var pt = this.abs();
|
||
if (pt.t < pm.t) {
|
||
if (q != null) q.fromInt(0);
|
||
if (r != null) this.copyTo(r);
|
||
return;
|
||
}
|
||
if (r == null) r = nbi();
|
||
var y = nbi(),
|
||
ts = this.s,
|
||
ms = m.s;
|
||
var nsh = this.DB - nbits(pm[pm.t - 1]); // normalize modulus
|
||
if (nsh > 0) {
|
||
pm.lShiftTo(nsh, y);
|
||
pt.lShiftTo(nsh, r);
|
||
} else {
|
||
pm.copyTo(y);
|
||
pt.copyTo(r);
|
||
}
|
||
var ys = y.t;
|
||
var y0 = y[ys - 1];
|
||
if (y0 == 0) return;
|
||
var yt = y0 * (1 << this.F1) + (ys > 1 ? y[ys - 2] >> this.F2 : 0);
|
||
var d1 = this.FV / yt,
|
||
d2 = (1 << this.F1) / yt,
|
||
e = 1 << this.F2;
|
||
var i = r.t,
|
||
j = i - ys,
|
||
t = q == null ? nbi() : q;
|
||
y.dlShiftTo(j, t);
|
||
if (r.compareTo(t) >= 0) {
|
||
r[r.t++] = 1;
|
||
r.subTo(t, r);
|
||
}
|
||
BigInteger.ONE.dlShiftTo(ys, t);
|
||
t.subTo(y, y); // "negative" y so we can replace sub with am later
|
||
while (y.t < ys) y[y.t++] = 0;
|
||
while (--j >= 0) {
|
||
// Estimate quotient digit
|
||
var qd =
|
||
r[--i] == y0
|
||
? this.DM
|
||
: Math.floor(r[i] * d1 + (r[i - 1] + e) * d2);
|
||
if ((r[i] += y.am(0, qd, r, j, 0, ys)) < qd) {
|
||
// Try it out
|
||
y.dlShiftTo(j, t);
|
||
r.subTo(t, r);
|
||
while (r[i] < --qd) r.subTo(t, r);
|
||
}
|
||
}
|
||
if (q != null) {
|
||
r.drShiftTo(ys, q);
|
||
if (ts != ms) BigInteger.ZERO.subTo(q, q);
|
||
}
|
||
r.t = ys;
|
||
r.clamp();
|
||
if (nsh > 0) r.rShiftTo(nsh, r); // Denormalize remainder
|
||
if (ts < 0) BigInteger.ZERO.subTo(r, r);
|
||
};
|
||
|
||
// (protected) return "-1/this % 2^DB"; useful for Mont. reduction
|
||
// justification:
|
||
// xy == 1 (mod m)
|
||
// xy = 1+km
|
||
// xy(2-xy) = (1+km)(1-km)
|
||
// x[y(2-xy)] = 1-k^2m^2
|
||
// x[y(2-xy)] == 1 (mod m^2)
|
||
// if y is 1/x mod m, then y(2-xy) is 1/x mod m^2
|
||
// should reduce x and y(2-xy) by m^2 at each step to keep size bounded.
|
||
// JS multiply "overflows" differently from C/C++, so care is needed here.
|
||
BigInteger.prototype.invDigit = function () {
|
||
if (this.t < 1) return 0;
|
||
var x = this[0];
|
||
if ((x & 1) == 0) return 0;
|
||
var y = x & 3; // y == 1/x mod 2^2
|
||
y = (y * (2 - (x & 0xf) * y)) & 0xf; // y == 1/x mod 2^4
|
||
y = (y * (2 - (x & 0xff) * y)) & 0xff; // y == 1/x mod 2^8
|
||
y = (y * (2 - (((x & 0xffff) * y) & 0xffff))) & 0xffff; // y == 1/x mod 2^16
|
||
// last step - calculate inverse mod DV directly;
|
||
// assumes 16 < DB <= 32 and assumes ability to handle 48-bit ints
|
||
y = (y * (2 - ((x * y) % this.DV))) % this.DV; // y == 1/x mod 2^dbits
|
||
// we really want the negative inverse, and -DV < y < DV
|
||
return y > 0 ? this.DV - y : -y;
|
||
};
|
||
|
||
// (protected) true iff this is even
|
||
BigInteger.prototype.isEven = function () {
|
||
return (this.t > 0 ? this[0] & 1 : this.s) == 0;
|
||
};
|
||
|
||
// (protected) this^e, e < 2^32, doing sqr and mul with "r" (HAC 14.79)
|
||
BigInteger.prototype.exp = function (e, z) {
|
||
if (e > 0xffffffff || e < 1) return BigInteger.ONE;
|
||
var r = nbi(),
|
||
r2 = nbi(),
|
||
g = z.convert(this),
|
||
i = nbits(e) - 1;
|
||
g.copyTo(r);
|
||
while (--i >= 0) {
|
||
z.sqrTo(r, r2);
|
||
if ((e & (1 << i)) > 0) z.mulTo(r2, g, r);
|
||
else {
|
||
var t = r;
|
||
r = r2;
|
||
r2 = t;
|
||
}
|
||
}
|
||
return z.revert(r);
|
||
};
|
||
|
||
// (public) return string representation in given radix
|
||
BigInteger.prototype.toString = function (b) {
|
||
if (this.s < 0) return "-" + this.negate().toString(b);
|
||
var k;
|
||
if (b == 16) k = 4;
|
||
else if (b == 8) k = 3;
|
||
else if (b == 2) k = 1;
|
||
else if (b == 32) k = 5;
|
||
else if (b == 4) k = 2;
|
||
else return this.toRadix(b);
|
||
var km = (1 << k) - 1,
|
||
d,
|
||
m = false,
|
||
r = "",
|
||
i = this.t;
|
||
var p = this.DB - ((i * this.DB) % k);
|
||
if (i-- > 0) {
|
||
if (p < this.DB && (d = this[i] >> p) > 0) {
|
||
m = true;
|
||
r = int2char(d);
|
||
}
|
||
while (i >= 0) {
|
||
if (p < k) {
|
||
d = (this[i] & ((1 << p) - 1)) << (k - p);
|
||
d |= this[--i] >> (p += this.DB - k);
|
||
} else {
|
||
d = (this[i] >> (p -= k)) & km;
|
||
if (p <= 0) {
|
||
p += this.DB;
|
||
--i;
|
||
}
|
||
}
|
||
if (d > 0) m = true;
|
||
if (m) r += int2char(d);
|
||
}
|
||
}
|
||
return m ? r : "0";
|
||
};
|
||
|
||
// (public) -this
|
||
BigInteger.prototype.negate = function () {
|
||
var r = nbi();
|
||
BigInteger.ZERO.subTo(this, r);
|
||
return r;
|
||
};
|
||
|
||
// (public) |this|
|
||
BigInteger.prototype.abs = function () {
|
||
return this.s < 0 ? this.negate() : this;
|
||
};
|
||
|
||
// (public) return + if this > a, - if this < a, 0 if equal
|
||
BigInteger.prototype.compareTo = function (a) {
|
||
var r = this.s - a.s;
|
||
if (r != 0) return r;
|
||
var i = this.t;
|
||
r = i - a.t;
|
||
if (r != 0) return this.s < 0 ? -r : r;
|
||
while (--i >= 0) if ((r = this[i] - a[i]) != 0) return r;
|
||
return 0;
|
||
};
|
||
|
||
// (public) return the number of bits in "this"
|
||
BigInteger.prototype.bitLength = function () {
|
||
if (this.t <= 0) return 0;
|
||
return (
|
||
this.DB * (this.t - 1) + nbits(this[this.t - 1] ^ (this.s & this.DM))
|
||
);
|
||
};
|
||
|
||
// (public) this mod a
|
||
BigInteger.prototype.mod = function (a) {
|
||
var r = nbi();
|
||
this.abs().divRemTo(a, null, r);
|
||
if (this.s < 0 && r.compareTo(BigInteger.ZERO) > 0) a.subTo(r, r);
|
||
return r;
|
||
};
|
||
|
||
// (public) this^e % m, 0 <= e < 2^32
|
||
BigInteger.prototype.modPowInt = function (e, m) {
|
||
var z;
|
||
if (e < 256 || m.isEven()) z = new Classic(m);
|
||
else z = new Montgomery(m);
|
||
return this.exp(e, z);
|
||
};
|
||
|
||
// "constants"
|
||
BigInteger.ZERO = nbv(0);
|
||
BigInteger.ONE = nbv(1);
|
||
|
||
// Copyright (c) 2005-2009 Tom Wu
|
||
// All Rights Reserved.
|
||
// See "LICENSE" for details.
|
||
// Extended JavaScript BN functions, required for RSA private ops.
|
||
// Version 1.1: new BigInteger("0", 10) returns "proper" zero
|
||
// Version 1.2: square() API, isProbablePrime fix
|
||
|
||
// return index of lowest 1-bit in x, x < 2^31
|
||
function lbit(x) {
|
||
if (x == 0) return -1;
|
||
var r = 0;
|
||
if ((x & 0xffff) == 0) {
|
||
x >>= 16;
|
||
r += 16;
|
||
}
|
||
if ((x & 0xff) == 0) {
|
||
x >>= 8;
|
||
r += 8;
|
||
}
|
||
if ((x & 0xf) == 0) {
|
||
x >>= 4;
|
||
r += 4;
|
||
}
|
||
if ((x & 3) == 0) {
|
||
x >>= 2;
|
||
r += 2;
|
||
}
|
||
if ((x & 1) == 0) ++r;
|
||
return r;
|
||
}
|
||
|
||
// return number of 1 bits in x
|
||
function cbit(x) {
|
||
var r = 0;
|
||
while (x != 0) {
|
||
x &= x - 1;
|
||
++r;
|
||
}
|
||
return r;
|
||
}
|
||
|
||
var lowprimes = [
|
||
2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53, 59, 61, 67,
|
||
71, 73, 79, 83, 89, 97, 101, 103, 107, 109, 113, 127, 131, 137, 139,
|
||
149, 151, 157, 163, 167, 173, 179, 181, 191, 193, 197, 199, 211, 223,
|
||
227, 229, 233, 239, 241, 251, 257, 263, 269, 271, 277, 281, 283, 293,
|
||
307, 311, 313, 317, 331, 337, 347, 349, 353, 359, 367, 373, 379, 383,
|
||
389, 397, 401, 409, 419, 421, 431, 433, 439, 443, 449, 457, 461, 463,
|
||
467, 479, 487, 491, 499, 503, 509, 521, 523, 541, 547, 557, 563, 569,
|
||
571, 577, 587, 593, 599, 601, 607, 613, 617, 619, 631, 641, 643, 647,
|
||
653, 659, 661, 673, 677, 683, 691, 701, 709, 719, 727, 733, 739, 743,
|
||
751, 757, 761, 769, 773, 787, 797, 809, 811, 821, 823, 827, 829, 839,
|
||
853, 857, 859, 863, 877, 881, 883, 887, 907, 911, 919, 929, 937, 941,
|
||
947, 953, 967, 971, 977, 983, 991, 997,
|
||
];
|
||
var lplim = (1 << 26) / lowprimes[lowprimes.length - 1];
|
||
|
||
// (protected) return x s.t. r^x < DV
|
||
BigInteger.prototype.chunkSize = function (r) {
|
||
return Math.floor((Math.LN2 * this.DB) / Math.log(r));
|
||
};
|
||
|
||
// (protected) convert to radix string
|
||
BigInteger.prototype.toRadix = function (b) {
|
||
if (b == null) b = 10;
|
||
if (this.signum() == 0 || b < 2 || b > 36) return "0";
|
||
var cs = this.chunkSize(b);
|
||
var a = Math.pow(b, cs);
|
||
var d = nbv(a),
|
||
y = nbi(),
|
||
z = nbi(),
|
||
r = "";
|
||
this.divRemTo(d, y, z);
|
||
while (y.signum() > 0) {
|
||
r = (a + z.intValue()).toString(b).substr(1) + r;
|
||
y.divRemTo(d, y, z);
|
||
}
|
||
return z.intValue().toString(b) + r;
|
||
};
|
||
|
||
// (protected) convert from radix string
|
||
BigInteger.prototype.fromRadix = function (s, b) {
|
||
this.fromInt(0);
|
||
if (b == null) b = 10;
|
||
var cs = this.chunkSize(b);
|
||
var d = Math.pow(b, cs),
|
||
mi = false,
|
||
j = 0,
|
||
w = 0;
|
||
for (var i = 0; i < s.length; ++i) {
|
||
var x = intAt(s, i);
|
||
if (x < 0) {
|
||
if (s.charAt(i) == "-" && this.signum() == 0) mi = true;
|
||
continue;
|
||
}
|
||
w = b * w + x;
|
||
if (++j >= cs) {
|
||
this.dMultiply(d);
|
||
this.dAddOffset(w, 0);
|
||
j = 0;
|
||
w = 0;
|
||
}
|
||
}
|
||
if (j > 0) {
|
||
this.dMultiply(Math.pow(b, j));
|
||
this.dAddOffset(w, 0);
|
||
}
|
||
if (mi) BigInteger.ZERO.subTo(this, this);
|
||
};
|
||
|
||
// (protected) alternate constructor
|
||
BigInteger.prototype.fromNumber = function (a, b, c) {
|
||
if ("number" == typeof b) {
|
||
// new BigInteger(int,int,RNG)
|
||
if (a < 2) this.fromInt(1);
|
||
else {
|
||
this.fromNumber(a, c);
|
||
if (!this.testBit(a - 1))
|
||
// force MSB set
|
||
this.bitwiseTo(BigInteger.ONE.shiftLeft(a - 1), op_or, this);
|
||
if (this.isEven()) this.dAddOffset(1, 0); // force odd
|
||
while (!this.isProbablePrime(b)) {
|
||
this.dAddOffset(2, 0);
|
||
if (this.bitLength() > a)
|
||
this.subTo(BigInteger.ONE.shiftLeft(a - 1), this);
|
||
}
|
||
}
|
||
} else {
|
||
// new BigInteger(int,RNG)
|
||
var x = new Array(),
|
||
t = a & 7;
|
||
x.length = (a >> 3) + 1;
|
||
b.nextBytes(x);
|
||
if (t > 0) x[0] &= (1 << t) - 1;
|
||
else x[0] = 0;
|
||
this.fromString(x, 256);
|
||
}
|
||
};
|
||
|
||
// (protected) r = this op a (bitwise)
|
||
BigInteger.prototype.bitwiseTo = function (a, op, r) {
|
||
var i,
|
||
f,
|
||
m = Math.min(a.t, this.t);
|
||
for (i = 0; i < m; ++i) r[i] = op(this[i], a[i]);
|
||
if (a.t < this.t) {
|
||
f = a.s & this.DM;
|
||
for (i = m; i < this.t; ++i) r[i] = op(this[i], f);
|
||
r.t = this.t;
|
||
} else {
|
||
f = this.s & this.DM;
|
||
for (i = m; i < a.t; ++i) r[i] = op(f, a[i]);
|
||
r.t = a.t;
|
||
}
|
||
r.s = op(this.s, a.s);
|
||
r.clamp();
|
||
};
|
||
|
||
// (protected) this op (1<<n)
|
||
BigInteger.prototype.changeBit = function (n, op) {
|
||
var r = BigInteger.ONE.shiftLeft(n);
|
||
this.bitwiseTo(r, op, r);
|
||
return r;
|
||
};
|
||
|
||
// (protected) r = this + a
|
||
BigInteger.prototype.addTo = function (a, r) {
|
||
var i = 0,
|
||
c = 0,
|
||
m = Math.min(a.t, this.t);
|
||
while (i < m) {
|
||
c += this[i] + a[i];
|
||
r[i++] = c & this.DM;
|
||
c >>= this.DB;
|
||
}
|
||
if (a.t < this.t) {
|
||
c += a.s;
|
||
while (i < this.t) {
|
||
c += this[i];
|
||
r[i++] = c & this.DM;
|
||
c >>= this.DB;
|
||
}
|
||
c += this.s;
|
||
} else {
|
||
c += this.s;
|
||
while (i < a.t) {
|
||
c += a[i];
|
||
r[i++] = c & this.DM;
|
||
c >>= this.DB;
|
||
}
|
||
c += a.s;
|
||
}
|
||
r.s = c < 0 ? -1 : 0;
|
||
if (c > 0) r[i++] = c;
|
||
else if (c < -1) r[i++] = this.DV + c;
|
||
r.t = i;
|
||
r.clamp();
|
||
};
|
||
|
||
// (protected) this *= n, this >= 0, 1 < n < DV
|
||
BigInteger.prototype.dMultiply = function (n) {
|
||
this[this.t] = this.am(0, n - 1, this, 0, 0, this.t);
|
||
++this.t;
|
||
this.clamp();
|
||
};
|
||
|
||
// (protected) this += n << w words, this >= 0
|
||
BigInteger.prototype.dAddOffset = function (n, w) {
|
||
if (n == 0) return;
|
||
while (this.t <= w) this[this.t++] = 0;
|
||
this[w] += n;
|
||
while (this[w] >= this.DV) {
|
||
this[w] -= this.DV;
|
||
if (++w >= this.t) this[this.t++] = 0;
|
||
++this[w];
|
||
}
|
||
};
|
||
|
||
// (protected) r = lower n words of "this * a", a.t <= n
|
||
// "this" should be the larger one if appropriate.
|
||
BigInteger.prototype.multiplyLowerTo = function (a, n, r) {
|
||
var i = Math.min(this.t + a.t, n);
|
||
r.s = 0; // assumes a,this >= 0
|
||
r.t = i;
|
||
while (i > 0) r[--i] = 0;
|
||
var j;
|
||
for (j = r.t - this.t; i < j; ++i)
|
||
r[i + this.t] = this.am(0, a[i], r, i, 0, this.t);
|
||
for (j = Math.min(a.t, n); i < j; ++i) this.am(0, a[i], r, i, 0, n - i);
|
||
r.clamp();
|
||
};
|
||
|
||
// (protected) r = "this * a" without lower n words, n > 0
|
||
// "this" should be the larger one if appropriate.
|
||
BigInteger.prototype.multiplyUpperTo = function (a, n, r) {
|
||
--n;
|
||
var i = (r.t = this.t + a.t - n);
|
||
r.s = 0; // assumes a,this >= 0
|
||
while (--i >= 0) r[i] = 0;
|
||
for (i = Math.max(n - this.t, 0); i < a.t; ++i)
|
||
r[this.t + i - n] = this.am(n - i, a[i], r, 0, 0, this.t + i - n);
|
||
r.clamp();
|
||
r.drShiftTo(1, r);
|
||
};
|
||
|
||
// (protected) this % n, n < 2^26
|
||
BigInteger.prototype.modInt = function (n) {
|
||
if (n <= 0) return 0;
|
||
var d = this.DV % n,
|
||
r = this.s < 0 ? n - 1 : 0;
|
||
if (this.t > 0)
|
||
if (d == 0) r = this[0] % n;
|
||
else for (var i = this.t - 1; i >= 0; --i) r = (d * r + this[i]) % n;
|
||
return r;
|
||
};
|
||
|
||
// (protected) true if probably prime (HAC 4.24, Miller-Rabin)
|
||
BigInteger.prototype.millerRabin = function (t) {
|
||
var n1 = this.subtract(BigInteger.ONE);
|
||
var k = n1.getLowestSetBit();
|
||
if (k <= 0) return false;
|
||
var r = n1.shiftRight(k);
|
||
t = (t + 1) >> 1;
|
||
if (t > lowprimes.length) t = lowprimes.length;
|
||
var a = nbi();
|
||
for (var i = 0; i < t; ++i) {
|
||
//Pick bases at random, instead of starting at 2
|
||
a.fromInt(lowprimes[Math.floor(Math.random() * lowprimes.length)]);
|
||
var y = a.modPow(r, this);
|
||
if (y.compareTo(BigInteger.ONE) != 0 && y.compareTo(n1) != 0) {
|
||
var j = 1;
|
||
while (j++ < k && y.compareTo(n1) != 0) {
|
||
y = y.modPowInt(2, this);
|
||
if (y.compareTo(BigInteger.ONE) == 0) return false;
|
||
}
|
||
if (y.compareTo(n1) != 0) return false;
|
||
}
|
||
}
|
||
return true;
|
||
};
|
||
|
||
// (public)
|
||
BigInteger.prototype.clone = function () {
|
||
var r = nbi();
|
||
this.copyTo(r);
|
||
return r;
|
||
};
|
||
|
||
// (public) return value as integer
|
||
BigInteger.prototype.intValue = function () {
|
||
if (this.s < 0) {
|
||
if (this.t == 1) return this[0] - this.DV;
|
||
else if (this.t == 0) return -1;
|
||
} else if (this.t == 1) return this[0];
|
||
else if (this.t == 0) return 0;
|
||
// assumes 16 < DB < 32
|
||
return ((this[1] & ((1 << (32 - this.DB)) - 1)) << this.DB) | this[0];
|
||
};
|
||
|
||
// (public) return value as byte
|
||
BigInteger.prototype.byteValue = function () {
|
||
return this.t == 0 ? this.s : (this[0] << 24) >> 24;
|
||
};
|
||
|
||
// (public) return value as short (assumes DB>=16)
|
||
BigInteger.prototype.shortValue = function () {
|
||
return this.t == 0 ? this.s : (this[0] << 16) >> 16;
|
||
};
|
||
|
||
// (public) 0 if this == 0, 1 if this > 0
|
||
BigInteger.prototype.signum = function () {
|
||
if (this.s < 0) return -1;
|
||
else if (this.t <= 0 || (this.t == 1 && this[0] <= 0)) return 0;
|
||
else return 1;
|
||
};
|
||
|
||
// (public) convert to bigendian byte array
|
||
BigInteger.prototype.toByteArray = function () {
|
||
var i = this.t,
|
||
r = new Array();
|
||
r[0] = this.s;
|
||
var p = this.DB - ((i * this.DB) % 8),
|
||
d,
|
||
k = 0;
|
||
if (i-- > 0) {
|
||
if (p < this.DB && (d = this[i] >> p) != (this.s & this.DM) >> p)
|
||
r[k++] = d | (this.s << (this.DB - p));
|
||
while (i >= 0) {
|
||
if (p < 8) {
|
||
d = (this[i] & ((1 << p) - 1)) << (8 - p);
|
||
d |= this[--i] >> (p += this.DB - 8);
|
||
} else {
|
||
d = (this[i] >> (p -= 8)) & 0xff;
|
||
if (p <= 0) {
|
||
p += this.DB;
|
||
--i;
|
||
}
|
||
}
|
||
if ((d & 0x80) != 0) d |= -256;
|
||
if (k == 0 && (this.s & 0x80) != (d & 0x80)) ++k;
|
||
if (k > 0 || d != this.s) r[k++] = d;
|
||
}
|
||
}
|
||
return r;
|
||
};
|
||
|
||
BigInteger.prototype.equals = function (a) {
|
||
return this.compareTo(a) == 0;
|
||
};
|
||
BigInteger.prototype.min = function (a) {
|
||
return this.compareTo(a) < 0 ? this : a;
|
||
};
|
||
BigInteger.prototype.max = function (a) {
|
||
return this.compareTo(a) > 0 ? this : a;
|
||
};
|
||
|
||
// (public) this & a
|
||
function op_and(x, y) {
|
||
return x & y;
|
||
}
|
||
BigInteger.prototype.and = function (a) {
|
||
var r = nbi();
|
||
this.bitwiseTo(a, op_and, r);
|
||
return r;
|
||
};
|
||
|
||
// (public) this | a
|
||
function op_or(x, y) {
|
||
return x | y;
|
||
}
|
||
BigInteger.prototype.or = function (a) {
|
||
var r = nbi();
|
||
this.bitwiseTo(a, op_or, r);
|
||
return r;
|
||
};
|
||
|
||
// (public) this ^ a
|
||
function op_xor(x, y) {
|
||
return x ^ y;
|
||
}
|
||
BigInteger.prototype.xor = function (a) {
|
||
var r = nbi();
|
||
this.bitwiseTo(a, op_xor, r);
|
||
return r;
|
||
};
|
||
|
||
// (public) this & ~a
|
||
function op_andnot(x, y) {
|
||
return x & ~y;
|
||
}
|
||
BigInteger.prototype.andNot = function (a) {
|
||
var r = nbi();
|
||
this.bitwiseTo(a, op_andnot, r);
|
||
return r;
|
||
};
|
||
|
||
// (public) ~this
|
||
BigInteger.prototype.not = function () {
|
||
var r = nbi();
|
||
for (var i = 0; i < this.t; ++i) r[i] = this.DM & ~this[i];
|
||
r.t = this.t;
|
||
r.s = ~this.s;
|
||
return r;
|
||
};
|
||
|
||
// (public) this << n
|
||
BigInteger.prototype.shiftLeft = function (n) {
|
||
var r = nbi();
|
||
if (n < 0) this.rShiftTo(-n, r);
|
||
else this.lShiftTo(n, r);
|
||
return r;
|
||
};
|
||
|
||
// (public) this >> n
|
||
BigInteger.prototype.shiftRight = function (n) {
|
||
var r = nbi();
|
||
if (n < 0) this.lShiftTo(-n, r);
|
||
else this.rShiftTo(n, r);
|
||
return r;
|
||
};
|
||
|
||
// (public) returns index of lowest 1-bit (or -1 if none)
|
||
BigInteger.prototype.getLowestSetBit = function () {
|
||
for (var i = 0; i < this.t; ++i)
|
||
if (this[i] != 0) return i * this.DB + lbit(this[i]);
|
||
if (this.s < 0) return this.t * this.DB;
|
||
return -1;
|
||
};
|
||
|
||
// (public) return number of set bits
|
||
BigInteger.prototype.bitCount = function () {
|
||
var r = 0,
|
||
x = this.s & this.DM;
|
||
for (var i = 0; i < this.t; ++i) r += cbit(this[i] ^ x);
|
||
return r;
|
||
};
|
||
|
||
// (public) true iff nth bit is set
|
||
BigInteger.prototype.testBit = function (n) {
|
||
var j = Math.floor(n / this.DB);
|
||
if (j >= this.t) return this.s != 0;
|
||
return (this[j] & (1 << n % this.DB)) != 0;
|
||
};
|
||
|
||
// (public) this | (1<<n)
|
||
BigInteger.prototype.setBit = function (n) {
|
||
return this.changeBit(n, op_or);
|
||
};
|
||
// (public) this & ~(1<<n)
|
||
BigInteger.prototype.clearBit = function (n) {
|
||
return this.changeBit(n, op_andnot);
|
||
};
|
||
// (public) this ^ (1<<n)
|
||
BigInteger.prototype.flipBit = function (n) {
|
||
return this.changeBit(n, op_xor);
|
||
};
|
||
// (public) this + a
|
||
BigInteger.prototype.add = function (a) {
|
||
var r = nbi();
|
||
this.addTo(a, r);
|
||
return r;
|
||
};
|
||
// (public) this - a
|
||
BigInteger.prototype.subtract = function (a) {
|
||
var r = nbi();
|
||
this.subTo(a, r);
|
||
return r;
|
||
};
|
||
// (public) this * a
|
||
BigInteger.prototype.multiply = function (a) {
|
||
var r = nbi();
|
||
this.multiplyTo(a, r);
|
||
return r;
|
||
};
|
||
// (public) this / a
|
||
BigInteger.prototype.divide = function (a) {
|
||
var r = nbi();
|
||
this.divRemTo(a, r, null);
|
||
return r;
|
||
};
|
||
// (public) this % a
|
||
BigInteger.prototype.remainder = function (a) {
|
||
var r = nbi();
|
||
this.divRemTo(a, null, r);
|
||
return r;
|
||
};
|
||
// (public) [this/a,this%a]
|
||
BigInteger.prototype.divideAndRemainder = function (a) {
|
||
var q = nbi(),
|
||
r = nbi();
|
||
this.divRemTo(a, q, r);
|
||
return new Array(q, r);
|
||
};
|
||
|
||
// (public) this^e % m (HAC 14.85)
|
||
BigInteger.prototype.modPow = function (e, m) {
|
||
var i = e.bitLength(),
|
||
k,
|
||
r = nbv(1),
|
||
z;
|
||
if (i <= 0) return r;
|
||
else if (i < 18) k = 1;
|
||
else if (i < 48) k = 3;
|
||
else if (i < 144) k = 4;
|
||
else if (i < 768) k = 5;
|
||
else k = 6;
|
||
if (i < 8) z = new Classic(m);
|
||
else if (m.isEven()) z = new Barrett(m);
|
||
else z = new Montgomery(m);
|
||
|
||
// precomputation
|
||
var g = new Array(),
|
||
n = 3,
|
||
k1 = k - 1,
|
||
km = (1 << k) - 1;
|
||
g[1] = z.convert(this);
|
||
if (k > 1) {
|
||
var g2 = nbi();
|
||
z.sqrTo(g[1], g2);
|
||
while (n <= km) {
|
||
g[n] = nbi();
|
||
z.mulTo(g2, g[n - 2], g[n]);
|
||
n += 2;
|
||
}
|
||
}
|
||
|
||
var j = e.t - 1,
|
||
w,
|
||
is1 = true,
|
||
r2 = nbi(),
|
||
t;
|
||
i = nbits(e[j]) - 1;
|
||
while (j >= 0) {
|
||
if (i >= k1) w = (e[j] >> (i - k1)) & km;
|
||
else {
|
||
w = (e[j] & ((1 << (i + 1)) - 1)) << (k1 - i);
|
||
if (j > 0) w |= e[j - 1] >> (this.DB + i - k1);
|
||
}
|
||
|
||
n = k;
|
||
while ((w & 1) == 0) {
|
||
w >>= 1;
|
||
--n;
|
||
}
|
||
if ((i -= n) < 0) {
|
||
i += this.DB;
|
||
--j;
|
||
}
|
||
if (is1) {
|
||
// ret == 1, don't bother squaring or multiplying it
|
||
g[w].copyTo(r);
|
||
is1 = false;
|
||
} else {
|
||
while (n > 1) {
|
||
z.sqrTo(r, r2);
|
||
z.sqrTo(r2, r);
|
||
n -= 2;
|
||
}
|
||
if (n > 0) z.sqrTo(r, r2);
|
||
else {
|
||
t = r;
|
||
r = r2;
|
||
r2 = t;
|
||
}
|
||
z.mulTo(r2, g[w], r);
|
||
}
|
||
|
||
while (j >= 0 && (e[j] & (1 << i)) == 0) {
|
||
z.sqrTo(r, r2);
|
||
t = r;
|
||
r = r2;
|
||
r2 = t;
|
||
if (--i < 0) {
|
||
i = this.DB - 1;
|
||
--j;
|
||
}
|
||
}
|
||
}
|
||
return z.revert(r);
|
||
};
|
||
|
||
// (public) 1/this % m (HAC 14.61)
|
||
BigInteger.prototype.modInverse = function (m) {
|
||
var ac = m.isEven();
|
||
if (this.signum() === 0) throw new Error("division by zero");
|
||
if ((this.isEven() && ac) || m.signum() == 0) return BigInteger.ZERO;
|
||
var u = m.clone(),
|
||
v = this.clone();
|
||
var a = nbv(1),
|
||
b = nbv(0),
|
||
c = nbv(0),
|
||
d = nbv(1);
|
||
while (u.signum() != 0) {
|
||
while (u.isEven()) {
|
||
u.rShiftTo(1, u);
|
||
if (ac) {
|
||
if (!a.isEven() || !b.isEven()) {
|
||
a.addTo(this, a);
|
||
b.subTo(m, b);
|
||
}
|
||
a.rShiftTo(1, a);
|
||
} else if (!b.isEven()) b.subTo(m, b);
|
||
b.rShiftTo(1, b);
|
||
}
|
||
while (v.isEven()) {
|
||
v.rShiftTo(1, v);
|
||
if (ac) {
|
||
if (!c.isEven() || !d.isEven()) {
|
||
c.addTo(this, c);
|
||
d.subTo(m, d);
|
||
}
|
||
c.rShiftTo(1, c);
|
||
} else if (!d.isEven()) d.subTo(m, d);
|
||
d.rShiftTo(1, d);
|
||
}
|
||
if (u.compareTo(v) >= 0) {
|
||
u.subTo(v, u);
|
||
if (ac) a.subTo(c, a);
|
||
b.subTo(d, b);
|
||
} else {
|
||
v.subTo(u, v);
|
||
if (ac) c.subTo(a, c);
|
||
d.subTo(b, d);
|
||
}
|
||
}
|
||
if (v.compareTo(BigInteger.ONE) != 0) return BigInteger.ZERO;
|
||
while (d.compareTo(m) >= 0) d.subTo(m, d);
|
||
while (d.signum() < 0) d.addTo(m, d);
|
||
return d;
|
||
};
|
||
|
||
// (public) this^e
|
||
BigInteger.prototype.pow = function (e) {
|
||
return this.exp(e, new NullExp());
|
||
};
|
||
|
||
// (public) gcd(this,a) (HAC 14.54)
|
||
BigInteger.prototype.gcd = function (a) {
|
||
var x = this.s < 0 ? this.negate() : this.clone();
|
||
var y = a.s < 0 ? a.negate() : a.clone();
|
||
if (x.compareTo(y) < 0) {
|
||
var t = x;
|
||
x = y;
|
||
y = t;
|
||
}
|
||
var i = x.getLowestSetBit(),
|
||
g = y.getLowestSetBit();
|
||
if (g < 0) return x;
|
||
if (i < g) g = i;
|
||
if (g > 0) {
|
||
x.rShiftTo(g, x);
|
||
y.rShiftTo(g, y);
|
||
}
|
||
while (x.signum() > 0) {
|
||
if ((i = x.getLowestSetBit()) > 0) x.rShiftTo(i, x);
|
||
if ((i = y.getLowestSetBit()) > 0) y.rShiftTo(i, y);
|
||
if (x.compareTo(y) >= 0) {
|
||
x.subTo(y, x);
|
||
x.rShiftTo(1, x);
|
||
} else {
|
||
y.subTo(x, y);
|
||
y.rShiftTo(1, y);
|
||
}
|
||
}
|
||
if (g > 0) y.lShiftTo(g, y);
|
||
return y;
|
||
};
|
||
|
||
// (public) test primality with certainty >= 1-.5^t
|
||
BigInteger.prototype.isProbablePrime = function (t) {
|
||
var i,
|
||
x = this.abs();
|
||
if (x.t == 1 && x[0] <= lowprimes[lowprimes.length - 1]) {
|
||
for (i = 0; i < lowprimes.length; ++i)
|
||
if (x[0] == lowprimes[i]) return true;
|
||
return false;
|
||
}
|
||
if (x.isEven()) return false;
|
||
i = 1;
|
||
while (i < lowprimes.length) {
|
||
var m = lowprimes[i],
|
||
j = i + 1;
|
||
while (j < lowprimes.length && m < lplim) m *= lowprimes[j++];
|
||
m = x.modInt(m);
|
||
while (i < j) if (m % lowprimes[i++] == 0) return false;
|
||
}
|
||
return x.millerRabin(t);
|
||
};
|
||
|
||
// JSBN-specific extension
|
||
|
||
// (public) this^2
|
||
BigInteger.prototype.square = function () {
|
||
var r = nbi();
|
||
this.squareTo(r);
|
||
return r;
|
||
};
|
||
|
||
// NOTE: BigInteger interfaces not implemented in jsbn:
|
||
// BigInteger(int signum, byte[] magnitude)
|
||
// double doubleValue()
|
||
// float floatValue()
|
||
// int hashCode()
|
||
// long longValue()
|
||
// static BigInteger valueOf(long val)
|
||
|
||
// Copyright Stephan Thomas (start) --- //
|
||
// https://raw.github.com/bitcoinjs/bitcoinjs-lib/07f9d55ccb6abd962efb6befdd37671f85ea4ff9/src/util.js
|
||
// BigInteger monkey patching
|
||
BigInteger.valueOf = nbv;
|
||
|
||
/**
|
||
* Returns a byte array representation of the big integer.
|
||
*
|
||
* This returns the absolute of the contained value in big endian
|
||
* form. A value of zero results in an empty array.
|
||
*/
|
||
BigInteger.prototype.toByteArrayUnsigned = function () {
|
||
var ba = this.abs().toByteArray();
|
||
if (ba.length) {
|
||
if (ba[0] == 0) {
|
||
ba = ba.slice(1);
|
||
}
|
||
return ba.map(function (v) {
|
||
return v < 0 ? v + 256 : v;
|
||
});
|
||
} else {
|
||
// Empty array, nothing to do
|
||
return ba;
|
||
}
|
||
};
|
||
|
||
/**
|
||
* Turns a byte array into a big integer.
|
||
*
|
||
* This function will interpret a byte array as a big integer in big
|
||
* endian notation and ignore leading zeros.
|
||
*/
|
||
BigInteger.fromByteArrayUnsigned = function (ba) {
|
||
if (!ba.length) {
|
||
return ba.valueOf(0);
|
||
} else if (ba[0] & 0x80) {
|
||
// Prepend a zero so the BigInteger class doesn't mistake this
|
||
// for a negative integer.
|
||
return new BigInteger([0].concat(ba));
|
||
} else {
|
||
return new BigInteger(ba);
|
||
}
|
||
};
|
||
|
||
/**
|
||
* Converts big integer to signed byte representation.
|
||
*
|
||
* The format for this value uses a the most significant bit as a sign
|
||
* bit. If the most significant bit is already occupied by the
|
||
* absolute value, an extra byte is prepended and the sign bit is set
|
||
* there.
|
||
*
|
||
* Examples:
|
||
*
|
||
* 0 => 0x00
|
||
* 1 => 0x01
|
||
* -1 => 0x81
|
||
* 127 => 0x7f
|
||
* -127 => 0xff
|
||
* 128 => 0x0080
|
||
* -128 => 0x8080
|
||
* 255 => 0x00ff
|
||
* -255 => 0x80ff
|
||
* 16300 => 0x3fac
|
||
* -16300 => 0xbfac
|
||
* 62300 => 0x00f35c
|
||
* -62300 => 0x80f35c
|
||
*/
|
||
BigInteger.prototype.toByteArraySigned = function () {
|
||
var val = this.abs().toByteArrayUnsigned();
|
||
var neg = this.compareTo(BigInteger.ZERO) < 0;
|
||
|
||
if (neg) {
|
||
if (val[0] & 0x80) {
|
||
val.unshift(0x80);
|
||
} else {
|
||
val[0] |= 0x80;
|
||
}
|
||
} else {
|
||
if (val[0] & 0x80) {
|
||
val.unshift(0x00);
|
||
}
|
||
}
|
||
|
||
return val;
|
||
};
|
||
|
||
/**
|
||
* Parse a signed big integer byte representation.
|
||
*
|
||
* For details on the format please see BigInteger.toByteArraySigned.
|
||
*/
|
||
BigInteger.fromByteArraySigned = function (ba) {
|
||
// Check for negative value
|
||
if (ba[0] & 0x80) {
|
||
// Remove sign bit
|
||
ba[0] &= 0x7f;
|
||
|
||
return BigInteger.fromByteArrayUnsigned(ba).negate();
|
||
} else {
|
||
return BigInteger.fromByteArrayUnsigned(ba);
|
||
}
|
||
};
|
||
// Copyright Stephan Thomas (end) --- //
|
||
|
||
// ****** REDUCTION ******* //
|
||
|
||
// Modular reduction using "classic" algorithm
|
||
var Classic = (window.Classic = function Classic(m) {
|
||
this.m = m;
|
||
});
|
||
Classic.prototype.convert = function (x) {
|
||
if (x.s < 0 || x.compareTo(this.m) >= 0) return x.mod(this.m);
|
||
else return x;
|
||
};
|
||
Classic.prototype.revert = function (x) {
|
||
return x;
|
||
};
|
||
Classic.prototype.reduce = function (x) {
|
||
x.divRemTo(this.m, null, x);
|
||
};
|
||
Classic.prototype.mulTo = function (x, y, r) {
|
||
x.multiplyTo(y, r);
|
||
this.reduce(r);
|
||
};
|
||
Classic.prototype.sqrTo = function (x, r) {
|
||
x.squareTo(r);
|
||
this.reduce(r);
|
||
};
|
||
|
||
// Montgomery reduction
|
||
var Montgomery = (window.Montgomery = function Montgomery(m) {
|
||
this.m = m;
|
||
this.mp = m.invDigit();
|
||
this.mpl = this.mp & 0x7fff;
|
||
this.mph = this.mp >> 15;
|
||
this.um = (1 << (m.DB - 15)) - 1;
|
||
this.mt2 = 2 * m.t;
|
||
});
|
||
// xR mod m
|
||
Montgomery.prototype.convert = function (x) {
|
||
var r = nbi();
|
||
x.abs().dlShiftTo(this.m.t, r);
|
||
r.divRemTo(this.m, null, r);
|
||
if (x.s < 0 && r.compareTo(BigInteger.ZERO) > 0) this.m.subTo(r, r);
|
||
return r;
|
||
};
|
||
// x/R mod m
|
||
Montgomery.prototype.revert = function (x) {
|
||
var r = nbi();
|
||
x.copyTo(r);
|
||
this.reduce(r);
|
||
return r;
|
||
};
|
||
// x = x/R mod m (HAC 14.32)
|
||
Montgomery.prototype.reduce = function (x) {
|
||
while (x.t <= this.mt2)
|
||
// pad x so am has enough room later
|
||
x[x.t++] = 0;
|
||
for (var i = 0; i < this.m.t; ++i) {
|
||
// faster way of calculating u0 = x[i]*mp mod DV
|
||
var j = x[i] & 0x7fff;
|
||
var u0 =
|
||
(j * this.mpl +
|
||
(((j * this.mph + (x[i] >> 15) * this.mpl) & this.um) << 15)) &
|
||
x.DM;
|
||
// use am to combine the multiply-shift-add into one call
|
||
j = i + this.m.t;
|
||
x[j] += this.m.am(0, u0, x, i, 0, this.m.t);
|
||
// propagate carry
|
||
while (x[j] >= x.DV) {
|
||
x[j] -= x.DV;
|
||
x[++j]++;
|
||
}
|
||
}
|
||
x.clamp();
|
||
x.drShiftTo(this.m.t, x);
|
||
if (x.compareTo(this.m) >= 0) x.subTo(this.m, x);
|
||
};
|
||
// r = "xy/R mod m"; x,y != r
|
||
Montgomery.prototype.mulTo = function (x, y, r) {
|
||
x.multiplyTo(y, r);
|
||
this.reduce(r);
|
||
};
|
||
// r = "x^2/R mod m"; x != r
|
||
Montgomery.prototype.sqrTo = function (x, r) {
|
||
x.squareTo(r);
|
||
this.reduce(r);
|
||
};
|
||
|
||
// A "null" reducer
|
||
var NullExp = (window.NullExp = function NullExp() {});
|
||
NullExp.prototype.convert = function (x) {
|
||
return x;
|
||
};
|
||
NullExp.prototype.revert = function (x) {
|
||
return x;
|
||
};
|
||
NullExp.prototype.mulTo = function (x, y, r) {
|
||
x.multiplyTo(y, r);
|
||
};
|
||
NullExp.prototype.sqrTo = function (x, r) {
|
||
x.squareTo(r);
|
||
};
|
||
|
||
// Barrett modular reduction
|
||
var Barrett = (window.Barrett = function Barrett(m) {
|
||
// setup Barrett
|
||
this.r2 = nbi();
|
||
this.q3 = nbi();
|
||
BigInteger.ONE.dlShiftTo(2 * m.t, this.r2);
|
||
this.mu = this.r2.divide(m);
|
||
this.m = m;
|
||
});
|
||
Barrett.prototype.convert = function (x) {
|
||
if (x.s < 0 || x.t > 2 * this.m.t) return x.mod(this.m);
|
||
else if (x.compareTo(this.m) < 0) return x;
|
||
else {
|
||
var r = nbi();
|
||
x.copyTo(r);
|
||
this.reduce(r);
|
||
return r;
|
||
}
|
||
};
|
||
Barrett.prototype.revert = function (x) {
|
||
return x;
|
||
};
|
||
// x = x mod m (HAC 14.42)
|
||
Barrett.prototype.reduce = function (x) {
|
||
x.drShiftTo(this.m.t - 1, this.r2);
|
||
if (x.t > this.m.t + 1) {
|
||
x.t = this.m.t + 1;
|
||
x.clamp();
|
||
}
|
||
this.mu.multiplyUpperTo(this.r2, this.m.t + 1, this.q3);
|
||
this.m.multiplyLowerTo(this.q3, this.m.t + 1, this.r2);
|
||
while (x.compareTo(this.r2) < 0) x.dAddOffset(1, this.m.t + 1);
|
||
x.subTo(this.r2, x);
|
||
while (x.compareTo(this.m) >= 0) x.subTo(this.m, x);
|
||
};
|
||
// r = x*y mod m; x,y != r
|
||
Barrett.prototype.mulTo = function (x, y, r) {
|
||
x.multiplyTo(y, r);
|
||
this.reduce(r);
|
||
};
|
||
// r = x^2 mod m; x != r
|
||
Barrett.prototype.sqrTo = function (x, r) {
|
||
x.squareTo(r);
|
||
this.reduce(r);
|
||
};
|
||
})();
|
||
|
||
// BigInteger interfaces not implemented in jsbn:
|
||
|
||
// BigInteger(int signum, byte[] magnitude)
|
||
// double doubleValue()
|
||
// float floatValue()
|
||
// int hashCode()
|
||
// long longValue()
|
||
// static BigInteger valueOf(long val)
|
||
|
||
//ellipticcurve.js
|
||
/*!
|
||
* Basic Javascript Elliptic Curve implementation
|
||
* Ported loosely from BouncyCastle's Java EC code
|
||
* Only Fp curves implemented for now
|
||
*
|
||
* Copyright Tom Wu, bitaddress.org BSD License.
|
||
* http://www-cs-students.stanford.edu/~tjw/jsbn/LICENSE
|
||
*/
|
||
(function () {
|
||
// Constructor function of Global EllipticCurve object
|
||
var ec = (window.EllipticCurve = function () {});
|
||
|
||
// ----------------
|
||
// ECFieldElementFp constructor
|
||
// q instanceof BigInteger
|
||
// x instanceof BigInteger
|
||
ec.FieldElementFp = function (q, x) {
|
||
this.x = x;
|
||
// TODO if(x.compareTo(q) >= 0) error
|
||
this.q = q;
|
||
};
|
||
|
||
ec.FieldElementFp.prototype.equals = function (other) {
|
||
if (other == this) return true;
|
||
return this.q.equals(other.q) && this.x.equals(other.x);
|
||
};
|
||
|
||
ec.FieldElementFp.prototype.toBigInteger = function () {
|
||
return this.x;
|
||
};
|
||
|
||
ec.FieldElementFp.prototype.negate = function () {
|
||
return new ec.FieldElementFp(this.q, this.x.negate().mod(this.q));
|
||
};
|
||
|
||
ec.FieldElementFp.prototype.add = function (b) {
|
||
return new ec.FieldElementFp(
|
||
this.q,
|
||
this.x.add(b.toBigInteger()).mod(this.q)
|
||
);
|
||
};
|
||
|
||
ec.FieldElementFp.prototype.subtract = function (b) {
|
||
return new ec.FieldElementFp(
|
||
this.q,
|
||
this.x.subtract(b.toBigInteger()).mod(this.q)
|
||
);
|
||
};
|
||
|
||
ec.FieldElementFp.prototype.multiply = function (b) {
|
||
return new ec.FieldElementFp(
|
||
this.q,
|
||
this.x.multiply(b.toBigInteger()).mod(this.q)
|
||
);
|
||
};
|
||
|
||
ec.FieldElementFp.prototype.square = function () {
|
||
return new ec.FieldElementFp(this.q, this.x.square().mod(this.q));
|
||
};
|
||
|
||
ec.FieldElementFp.prototype.divide = function (b) {
|
||
return new ec.FieldElementFp(
|
||
this.q,
|
||
this.x.multiply(b.toBigInteger().modInverse(this.q)).mod(this.q)
|
||
);
|
||
};
|
||
|
||
ec.FieldElementFp.prototype.getByteLength = function () {
|
||
return Math.floor((this.toBigInteger().bitLength() + 7) / 8);
|
||
};
|
||
|
||
// D.1.4 91
|
||
/**
|
||
* return a sqrt root - the routine verifies that the calculation
|
||
* returns the right value - if none exists it returns null.
|
||
*
|
||
* Copyright (c) 2000 - 2011 The Legion Of The Bouncy Castle (http://www.bouncycastle.org)
|
||
* Ported to JavaScript by bitaddress.org
|
||
*/
|
||
ec.FieldElementFp.prototype.sqrt = function () {
|
||
if (!this.q.testBit(0)) throw new Error("even value of q");
|
||
|
||
// p mod 4 == 3
|
||
if (this.q.testBit(1)) {
|
||
// z = g^(u+1) + p, p = 4u + 3
|
||
var z = new ec.FieldElementFp(
|
||
this.q,
|
||
this.x.modPow(this.q.shiftRight(2).add(BigInteger.ONE), this.q)
|
||
);
|
||
return z.square().equals(this) ? z : null;
|
||
}
|
||
|
||
// p mod 4 == 1
|
||
var qMinusOne = this.q.subtract(BigInteger.ONE);
|
||
var legendreExponent = qMinusOne.shiftRight(1);
|
||
if (!this.x.modPow(legendreExponent, this.q).equals(BigInteger.ONE))
|
||
return null;
|
||
var u = qMinusOne.shiftRight(2);
|
||
var k = u.shiftLeft(1).add(BigInteger.ONE);
|
||
var Q = this.x;
|
||
var fourQ = Q.shiftLeft(2).mod(this.q);
|
||
var U, V;
|
||
|
||
do {
|
||
var rand = new SecureRandom();
|
||
var P;
|
||
do {
|
||
P = new BigInteger(this.q.bitLength(), rand);
|
||
} while (
|
||
P.compareTo(this.q) >= 0 ||
|
||
!P.multiply(P)
|
||
.subtract(fourQ)
|
||
.modPow(legendreExponent, this.q)
|
||
.equals(qMinusOne)
|
||
);
|
||
|
||
var result = ec.FieldElementFp.fastLucasSequence(this.q, P, Q, k);
|
||
|
||
U = result[0];
|
||
V = result[1];
|
||
if (V.multiply(V).mod(this.q).equals(fourQ)) {
|
||
// Integer division by 2, mod q
|
||
if (V.testBit(0)) {
|
||
V = V.add(this.q);
|
||
}
|
||
V = V.shiftRight(1);
|
||
return new ec.FieldElementFp(this.q, V);
|
||
}
|
||
} while (U.equals(BigInteger.ONE) || U.equals(qMinusOne));
|
||
|
||
return null;
|
||
};
|
||
/*!
|
||
* Crypto-JS 2.5.4 BlockModes.js
|
||
* contribution from Simon Greatrix
|
||
*/
|
||
|
||
(function (C) {
|
||
// Create pad namespace
|
||
var C_pad = (C.pad = {});
|
||
|
||
// Calculate the number of padding bytes required.
|
||
function _requiredPadding(cipher, message) {
|
||
var blockSizeInBytes = cipher._blocksize * 4;
|
||
var reqd = blockSizeInBytes - (message.length % blockSizeInBytes);
|
||
return reqd;
|
||
}
|
||
|
||
// Remove padding when the final byte gives the number of padding bytes.
|
||
var _unpadLength = function (cipher, message, alg, padding) {
|
||
var pad = message.pop();
|
||
if (pad == 0) {
|
||
throw new Error(
|
||
"Invalid zero-length padding specified for " +
|
||
alg +
|
||
". Wrong cipher specification or key used?"
|
||
);
|
||
}
|
||
var maxPad = cipher._blocksize * 4;
|
||
if (pad > maxPad) {
|
||
throw new Error(
|
||
"Invalid padding length of " +
|
||
pad +
|
||
" specified for " +
|
||
alg +
|
||
". Wrong cipher specification or key used?"
|
||
);
|
||
}
|
||
for (var i = 1; i < pad; i++) {
|
||
var b = message.pop();
|
||
if (padding != undefined && padding != b) {
|
||
throw new Error(
|
||
"Invalid padding byte of 0x" +
|
||
b.toString(16) +
|
||
" specified for " +
|
||
alg +
|
||
". Wrong cipher specification or key used?"
|
||
);
|
||
}
|
||
}
|
||
};
|
||
|
||
// No-operation padding, used for stream ciphers
|
||
C_pad.NoPadding = {
|
||
pad: function (cipher, message) {},
|
||
unpad: function (cipher, message) {},
|
||
};
|
||
|
||
// Zero Padding.
|
||
//
|
||
// If the message is not an exact number of blocks, the final block is
|
||
// completed with 0x00 bytes. There is no unpadding.
|
||
C_pad.ZeroPadding = {
|
||
pad: function (cipher, message) {
|
||
var blockSizeInBytes = cipher._blocksize * 4;
|
||
var reqd = message.length % blockSizeInBytes;
|
||
if (reqd != 0) {
|
||
for (reqd = blockSizeInBytes - reqd; reqd > 0; reqd--) {
|
||
message.push(0x00);
|
||
}
|
||
}
|
||
},
|
||
|
||
unpad: function (cipher, message) {
|
||
while (message[message.length - 1] == 0) {
|
||
message.pop();
|
||
}
|
||
},
|
||
};
|
||
|
||
// ISO/IEC 7816-4 padding.
|
||
//
|
||
// Pads the plain text with an 0x80 byte followed by as many 0x00
|
||
// bytes are required to complete the block.
|
||
C_pad.iso7816 = {
|
||
pad: function (cipher, message) {
|
||
var reqd = _requiredPadding(cipher, message);
|
||
message.push(0x80);
|
||
for (; reqd > 1; reqd--) {
|
||
message.push(0x00);
|
||
}
|
||
},
|
||
|
||
unpad: function (cipher, message) {
|
||
var padLength;
|
||
for (
|
||
padLength = cipher._blocksize * 4;
|
||
padLength > 0;
|
||
padLength--
|
||
) {
|
||
var b = message.pop();
|
||
if (b == 0x80) return;
|
||
if (b != 0x00) {
|
||
throw new Error(
|
||
"ISO-7816 padding byte must be 0, not 0x" +
|
||
b.toString(16) +
|
||
". Wrong cipher specification or key used?"
|
||
);
|
||
}
|
||
}
|
||
throw new Error(
|
||
"ISO-7816 padded beyond cipher block size. Wrong cipher specification or key used?"
|
||
);
|
||
},
|
||
};
|
||
|
||
// ANSI X.923 padding
|
||
//
|
||
// The final block is padded with zeros except for the last byte of the
|
||
// last block which contains the number of padding bytes.
|
||
C_pad.ansix923 = {
|
||
pad: function (cipher, message) {
|
||
var reqd = _requiredPadding(cipher, message);
|
||
for (var i = 1; i < reqd; i++) {
|
||
message.push(0x00);
|
||
}
|
||
message.push(reqd);
|
||
},
|
||
|
||
unpad: function (cipher, message) {
|
||
_unpadLength(cipher, message, "ANSI X.923", 0);
|
||
},
|
||
};
|
||
|
||
// ISO 10126
|
||
//
|
||
// The final block is padded with random bytes except for the last
|
||
// byte of the last block which contains the number of padding bytes.
|
||
C_pad.iso10126 = {
|
||
pad: function (cipher, message) {
|
||
var reqd = _requiredPadding(cipher, message);
|
||
for (var i = 1; i < reqd; i++) {
|
||
message.push(Math.floor(Math.random() * 256));
|
||
}
|
||
message.push(reqd);
|
||
},
|
||
|
||
unpad: function (cipher, message) {
|
||
_unpadLength(cipher, message, "ISO 10126", undefined);
|
||
},
|
||
};
|
||
|
||
// PKCS7 padding
|
||
//
|
||
// PKCS7 is described in RFC 5652. Padding is in whole bytes. The
|
||
// value of each added byte is the number of bytes that are added,
|
||
// i.e. N bytes, each of value N are added.
|
||
C_pad.pkcs7 = {
|
||
pad: function (cipher, message) {
|
||
var reqd = _requiredPadding(cipher, message);
|
||
for (var i = 0; i < reqd; i++) {
|
||
message.push(reqd);
|
||
}
|
||
},
|
||
|
||
unpad: function (cipher, message) {
|
||
_unpadLength(
|
||
cipher,
|
||
message,
|
||
"PKCS 7",
|
||
message[message.length - 1]
|
||
);
|
||
},
|
||
};
|
||
|
||
// Create mode namespace
|
||
var C_mode = (C.mode = {});
|
||
|
||
/**
|
||
* Mode base "class".
|
||
*/
|
||
var Mode = (C_mode.Mode = function (padding) {
|
||
if (padding) {
|
||
this._padding = padding;
|
||
}
|
||
});
|
||
|
||
Mode.prototype = {
|
||
encrypt: function (cipher, m, iv) {
|
||
this._padding.pad(cipher, m);
|
||
this._doEncrypt(cipher, m, iv);
|
||
},
|
||
|
||
decrypt: function (cipher, m, iv) {
|
||
this._doDecrypt(cipher, m, iv);
|
||
this._padding.unpad(cipher, m);
|
||
},
|
||
|
||
// Default padding
|
||
_padding: C_pad.iso7816,
|
||
};
|
||
|
||
/**
|
||
* Electronic Code Book mode.
|
||
*
|
||
* ECB applies the cipher directly against each block of the input.
|
||
*
|
||
* ECB does not require an initialization vector.
|
||
*/
|
||
var ECB = (C_mode.ECB = function () {
|
||
// Call parent constructor
|
||
Mode.apply(this, arguments);
|
||
});
|
||
|
||
// Inherit from Mode
|
||
var ECB_prototype = (ECB.prototype = new Mode());
|
||
|
||
// Concrete steps for Mode template
|
||
ECB_prototype._doEncrypt = function (cipher, m, iv) {
|
||
var blockSizeInBytes = cipher._blocksize * 4;
|
||
// Encrypt each block
|
||
for (var offset = 0; offset < m.length; offset += blockSizeInBytes) {
|
||
cipher._encryptblock(m, offset);
|
||
}
|
||
};
|
||
ECB_prototype._doDecrypt = function (cipher, c, iv) {
|
||
var blockSizeInBytes = cipher._blocksize * 4;
|
||
// Decrypt each block
|
||
for (var offset = 0; offset < c.length; offset += blockSizeInBytes) {
|
||
cipher._decryptblock(c, offset);
|
||
}
|
||
};
|
||
|
||
// ECB never uses an IV
|
||
ECB_prototype.fixOptions = function (options) {
|
||
options.iv = [];
|
||
};
|
||
|
||
/**
|
||
* Cipher block chaining
|
||
*
|
||
* The first block is XORed with the IV. Subsequent blocks are XOR with the
|
||
* previous cipher output.
|
||
*/
|
||
var CBC = (C_mode.CBC = function () {
|
||
// Call parent constructor
|
||
Mode.apply(this, arguments);
|
||
});
|
||
|
||
// Inherit from Mode
|
||
var CBC_prototype = (CBC.prototype = new Mode());
|
||
|
||
// Concrete steps for Mode template
|
||
CBC_prototype._doEncrypt = function (cipher, m, iv) {
|
||
var blockSizeInBytes = cipher._blocksize * 4;
|
||
|
||
// Encrypt each block
|
||
for (var offset = 0; offset < m.length; offset += blockSizeInBytes) {
|
||
if (offset == 0) {
|
||
// XOR first block using IV
|
||
for (var i = 0; i < blockSizeInBytes; i++) m[i] ^= iv[i];
|
||
} else {
|
||
// XOR this block using previous crypted block
|
||
for (var i = 0; i < blockSizeInBytes; i++)
|
||
m[offset + i] ^= m[offset + i - blockSizeInBytes];
|
||
}
|
||
// Encrypt block
|
||
cipher._encryptblock(m, offset);
|
||
}
|
||
};
|
||
CBC_prototype._doDecrypt = function (cipher, c, iv) {
|
||
var blockSizeInBytes = cipher._blocksize * 4;
|
||
|
||
// At the start, the previously crypted block is the IV
|
||
var prevCryptedBlock = iv;
|
||
|
||
// Decrypt each block
|
||
for (var offset = 0; offset < c.length; offset += blockSizeInBytes) {
|
||
// Save this crypted block
|
||
var thisCryptedBlock = c.slice(offset, offset + blockSizeInBytes);
|
||
// Decrypt block
|
||
cipher._decryptblock(c, offset);
|
||
// XOR decrypted block using previous crypted block
|
||
for (var i = 0; i < blockSizeInBytes; i++) {
|
||
c[offset + i] ^= prevCryptedBlock[i];
|
||
}
|
||
prevCryptedBlock = thisCryptedBlock;
|
||
}
|
||
};
|
||
|
||
/**
|
||
* Cipher feed back
|
||
*
|
||
* The cipher output is XORed with the plain text to produce the cipher output,
|
||
* which is then fed back into the cipher to produce a bit pattern to XOR the
|
||
* next block with.
|
||
*
|
||
* This is a stream cipher mode and does not require padding.
|
||
*/
|
||
var CFB = (C_mode.CFB = function () {
|
||
// Call parent constructor
|
||
Mode.apply(this, arguments);
|
||
});
|
||
|
||
// Inherit from Mode
|
||
var CFB_prototype = (CFB.prototype = new Mode());
|
||
|
||
// Override padding
|
||
CFB_prototype._padding = C_pad.NoPadding;
|
||
|
||
// Concrete steps for Mode template
|
||
CFB_prototype._doEncrypt = function (cipher, m, iv) {
|
||
var blockSizeInBytes = cipher._blocksize * 4,
|
||
keystream = iv.slice(0);
|
||
|
||
// Encrypt each byte
|
||
for (var i = 0; i < m.length; i++) {
|
||
var j = i % blockSizeInBytes;
|
||
if (j == 0) cipher._encryptblock(keystream, 0);
|
||
|
||
m[i] ^= keystream[j];
|
||
keystream[j] = m[i];
|
||
}
|
||
};
|
||
CFB_prototype._doDecrypt = function (cipher, c, iv) {
|
||
var blockSizeInBytes = cipher._blocksize * 4,
|
||
keystream = iv.slice(0);
|
||
|
||
// Encrypt each byte
|
||
for (var i = 0; i < c.length; i++) {
|
||
var j = i % blockSizeInBytes;
|
||
if (j == 0) cipher._encryptblock(keystream, 0);
|
||
|
||
var b = c[i];
|
||
c[i] ^= keystream[j];
|
||
keystream[j] = b;
|
||
}
|
||
};
|
||
|
||
/**
|
||
* Output feed back
|
||
*
|
||
* The cipher repeatedly encrypts its own output. The output is XORed with the
|
||
* plain text to produce the cipher text.
|
||
*
|
||
* This is a stream cipher mode and does not require padding.
|
||
*/
|
||
var OFB = (C_mode.OFB = function () {
|
||
// Call parent constructor
|
||
Mode.apply(this, arguments);
|
||
});
|
||
|
||
// Inherit from Mode
|
||
var OFB_prototype = (OFB.prototype = new Mode());
|
||
|
||
// Override padding
|
||
OFB_prototype._padding = C_pad.NoPadding;
|
||
|
||
// Concrete steps for Mode template
|
||
OFB_prototype._doEncrypt = function (cipher, m, iv) {
|
||
var blockSizeInBytes = cipher._blocksize * 4,
|
||
keystream = iv.slice(0);
|
||
|
||
// Encrypt each byte
|
||
for (var i = 0; i < m.length; i++) {
|
||
// Generate keystream
|
||
if (i % blockSizeInBytes == 0) cipher._encryptblock(keystream, 0);
|
||
|
||
// Encrypt byte
|
||
m[i] ^= keystream[i % blockSizeInBytes];
|
||
}
|
||
};
|
||
OFB_prototype._doDecrypt = OFB_prototype._doEncrypt;
|
||
|
||
/**
|
||
* Counter
|
||
* @author Gergely Risko
|
||
*
|
||
* After every block the last 4 bytes of the IV is increased by one
|
||
* with carry and that IV is used for the next block.
|
||
*
|
||
* This is a stream cipher mode and does not require padding.
|
||
*/
|
||
var CTR = (C_mode.CTR = function () {
|
||
// Call parent constructor
|
||
Mode.apply(this, arguments);
|
||
});
|
||
|
||
// Inherit from Mode
|
||
var CTR_prototype = (CTR.prototype = new Mode());
|
||
|
||
// Override padding
|
||
CTR_prototype._padding = C_pad.NoPadding;
|
||
|
||
CTR_prototype._doEncrypt = function (cipher, m, iv) {
|
||
var blockSizeInBytes = cipher._blocksize * 4;
|
||
var counter = iv.slice(0);
|
||
|
||
for (var i = 0; i < m.length; ) {
|
||
// do not lose iv
|
||
var keystream = counter.slice(0);
|
||
|
||
// Generate keystream for next block
|
||
cipher._encryptblock(keystream, 0);
|
||
|
||
// XOR keystream with block
|
||
for (var j = 0; i < m.length && j < blockSizeInBytes; j++, i++) {
|
||
m[i] ^= keystream[j];
|
||
}
|
||
|
||
// Increase counter
|
||
if (++counter[blockSizeInBytes - 1] == 256) {
|
||
counter[blockSizeInBytes - 1] = 0;
|
||
if (++counter[blockSizeInBytes - 2] == 256) {
|
||
counter[blockSizeInBytes - 2] = 0;
|
||
if (++counter[blockSizeInBytes - 3] == 256) {
|
||
counter[blockSizeInBytes - 3] = 0;
|
||
++counter[blockSizeInBytes - 4];
|
||
}
|
||
}
|
||
}
|
||
}
|
||
};
|
||
CTR_prototype._doDecrypt = CTR_prototype._doEncrypt;
|
||
})(Crypto);
|
||
|
||
/*!
|
||
* Crypto-JS v2.5.4 PBKDF2.js
|
||
* http://code.google.com/p/crypto-js/
|
||
* Copyright (c) 2009-2013, Jeff Mott. All rights reserved.
|
||
* http://code.google.com/p/crypto-js/wiki/License
|
||
*/
|
||
(function () {
|
||
// Shortcuts
|
||
var C = Crypto,
|
||
util = C.util,
|
||
charenc = C.charenc,
|
||
UTF8 = charenc.UTF8,
|
||
Binary = charenc.Binary;
|
||
|
||
C.PBKDF2 = function (password, salt, keylen, options) {
|
||
// Convert to byte arrays
|
||
if (password.constructor == String)
|
||
password = UTF8.stringToBytes(password);
|
||
if (salt.constructor == String) salt = UTF8.stringToBytes(salt);
|
||
/* else, assume byte arrays already */
|
||
|
||
// Defaults
|
||
var hasher = (options && options.hasher) || C.SHA1,
|
||
iterations = (options && options.iterations) || 1;
|
||
|
||
// Pseudo-random function
|
||
function PRF(password, salt) {
|
||
return C.HMAC(hasher, salt, password, {
|
||
asBytes: true,
|
||
});
|
||
}
|
||
|
||
// Generate key
|
||
var derivedKeyBytes = [],
|
||
blockindex = 1;
|
||
while (derivedKeyBytes.length < keylen) {
|
||
var block = PRF(
|
||
password,
|
||
salt.concat(util.wordsToBytes([blockindex]))
|
||
);
|
||
for (var u = block, i = 1; i < iterations; i++) {
|
||
u = PRF(password, u);
|
||
for (var j = 0; j < block.length; j++) block[j] ^= u[j];
|
||
}
|
||
derivedKeyBytes = derivedKeyBytes.concat(block);
|
||
blockindex++;
|
||
}
|
||
|
||
// Truncate excess bytes
|
||
derivedKeyBytes.length = keylen;
|
||
|
||
return options && options.asBytes
|
||
? derivedKeyBytes
|
||
: options && options.asString
|
||
? Binary.bytesToString(derivedKeyBytes)
|
||
: util.bytesToHex(derivedKeyBytes);
|
||
};
|
||
})();
|
||
|
||
/*
|
||
* Copyright (c) 2010-2011 Intalio Pte, All Rights Reserved
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in
|
||
* all copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||
* THE SOFTWARE.
|
||
*/
|
||
// https://github.com/cheongwy/node-scrypt-js
|
||
(function () {
|
||
var MAX_VALUE = 2147483647;
|
||
var workerUrl = null;
|
||
|
||
//function scrypt(byte[] passwd, byte[] salt, int N, int r, int p, int dkLen)
|
||
/*
|
||
* N = Cpu cost
|
||
* r = Memory cost
|
||
* p = parallelization cost
|
||
*
|
||
*/
|
||
window.Crypto_scrypt = function (
|
||
passwd,
|
||
salt,
|
||
N,
|
||
r,
|
||
p,
|
||
dkLen,
|
||
callback
|
||
) {
|
||
if (N == 0 || (N & (N - 1)) != 0)
|
||
throw Error("N must be > 0 and a power of 2");
|
||
|
||
if (N > MAX_VALUE / 128 / r) throw Error("Parameter N is too large");
|
||
if (r > MAX_VALUE / 128 / p) throw Error("Parameter r is too large");
|
||
|
||
var PBKDF2_opts = {
|
||
iterations: 1,
|
||
hasher: Crypto.SHA256,
|
||
asBytes: true,
|
||
};
|
||
|
||
var B = Crypto.PBKDF2(passwd, salt, p * 128 * r, PBKDF2_opts);
|
||
|
||
try {
|
||
var i = 0;
|
||
var worksDone = 0;
|
||
var makeWorker = function () {
|
||
if (!workerUrl) {
|
||
var code = "(" + scryptCore.toString() + ")()";
|
||
var blob;
|
||
try {
|
||
blob = new Blob([code], {
|
||
type: "text/javascript",
|
||
});
|
||
} catch (e) {
|
||
window.BlobBuilder =
|
||
window.BlobBuilder ||
|
||
window.WebKitBlobBuilder ||
|
||
window.MozBlobBuilder ||
|
||
window.MSBlobBuilder;
|
||
blob = new BlobBuilder();
|
||
blob.append(code);
|
||
blob = blob.getBlob("text/javascript");
|
||
}
|
||
workerUrl = URL.createObjectURL(blob);
|
||
}
|
||
var worker = new Worker(workerUrl);
|
||
worker.onmessage = function (event) {
|
||
var Bi = event.data[0],
|
||
Bslice = event.data[1];
|
||
worksDone++;
|
||
|
||
if (i < p) {
|
||
worker.postMessage([N, r, p, B, i++]);
|
||
}
|
||
|
||
var length = Bslice.length,
|
||
destPos = Bi * 128 * r,
|
||
srcPos = 0;
|
||
while (length--) {
|
||
B[destPos++] = Bslice[srcPos++];
|
||
}
|
||
|
||
if (worksDone == p) {
|
||
callback(Crypto.PBKDF2(passwd, B, dkLen, PBKDF2_opts));
|
||
}
|
||
};
|
||
return worker;
|
||
};
|
||
var workers = [makeWorker(), makeWorker()];
|
||
workers[0].postMessage([N, r, p, B, i++]);
|
||
if (p > 1) {
|
||
workers[1].postMessage([N, r, p, B, i++]);
|
||
}
|
||
} catch (e) {
|
||
window.setTimeout(function () {
|
||
scryptCore();
|
||
callback(Crypto.PBKDF2(passwd, B, dkLen, PBKDF2_opts));
|
||
}, 0);
|
||
}
|
||
|
||
// using this function to enclose everything needed to create a worker (but also invokable directly for synchronous use)
|
||
function scryptCore() {
|
||
var XY = [],
|
||
V = [];
|
||
|
||
if (typeof B === "undefined") {
|
||
onmessage = function (event) {
|
||
var data = event.data;
|
||
var N = data[0],
|
||
r = data[1],
|
||
p = data[2],
|
||
B = data[3],
|
||
i = data[4];
|
||
|
||
var Bslice = [];
|
||
arraycopy32(B, i * 128 * r, Bslice, 0, 128 * r);
|
||
smix(Bslice, 0, r, N, V, XY);
|
||
|
||
postMessage([i, Bslice]);
|
||
};
|
||
} else {
|
||
for (var i = 0; i < p; i++) {
|
||
smix(B, i * 128 * r, r, N, V, XY);
|
||
}
|
||
}
|
||
|
||
function smix(B, Bi, r, N, V, XY) {
|
||
var Xi = 0;
|
||
var Yi = 128 * r;
|
||
var i;
|
||
|
||
arraycopy32(B, Bi, XY, Xi, Yi);
|
||
|
||
for (i = 0; i < N; i++) {
|
||
arraycopy32(XY, Xi, V, i * Yi, Yi);
|
||
blockmix_salsa8(XY, Xi, Yi, r);
|
||
}
|
||
|
||
for (i = 0; i < N; i++) {
|
||
var j = integerify(XY, Xi, r) & (N - 1);
|
||
blockxor(V, j * Yi, XY, Xi, Yi);
|
||
blockmix_salsa8(XY, Xi, Yi, r);
|
||
}
|
||
|
||
arraycopy32(XY, Xi, B, Bi, Yi);
|
||
}
|
||
|
||
function blockmix_salsa8(BY, Bi, Yi, r) {
|
||
var X = [];
|
||
var i;
|
||
|
||
arraycopy32(BY, Bi + (2 * r - 1) * 64, X, 0, 64);
|
||
|
||
for (i = 0; i < 2 * r; i++) {
|
||
blockxor(BY, i * 64, X, 0, 64);
|
||
salsa20_8(X);
|
||
arraycopy32(X, 0, BY, Yi + i * 64, 64);
|
||
}
|
||
|
||
for (i = 0; i < r; i++) {
|
||
arraycopy32(BY, Yi + i * 2 * 64, BY, Bi + i * 64, 64);
|
||
}
|
||
|
||
for (i = 0; i < r; i++) {
|
||
arraycopy32(
|
||
BY,
|
||
Yi + (i * 2 + 1) * 64,
|
||
BY,
|
||
Bi + (i + r) * 64,
|
||
64
|
||
);
|
||
}
|
||
}
|
||
|
||
function R(a, b) {
|
||
return (a << b) | (a >>> (32 - b));
|
||
}
|
||
|
||
function salsa20_8(B) {
|
||
var B32 = new Array(32);
|
||
var x = new Array(32);
|
||
var i;
|
||
|
||
for (i = 0; i < 16; i++) {
|
||
B32[i] = (B[i * 4 + 0] & 0xff) << 0;
|
||
B32[i] |= (B[i * 4 + 1] & 0xff) << 8;
|
||
B32[i] |= (B[i * 4 + 2] & 0xff) << 16;
|
||
B32[i] |= (B[i * 4 + 3] & 0xff) << 24;
|
||
}
|
||
|
||
arraycopy(B32, 0, x, 0, 16);
|
||
|
||
for (i = 8; i > 0; i -= 2) {
|
||
x[4] ^= R(x[0] + x[12], 7);
|
||
x[8] ^= R(x[4] + x[0], 9);
|
||
x[12] ^= R(x[8] + x[4], 13);
|
||
x[0] ^= R(x[12] + x[8], 18);
|
||
x[9] ^= R(x[5] + x[1], 7);
|
||
x[13] ^= R(x[9] + x[5], 9);
|
||
x[1] ^= R(x[13] + x[9], 13);
|
||
x[5] ^= R(x[1] + x[13], 18);
|
||
x[14] ^= R(x[10] + x[6], 7);
|
||
x[2] ^= R(x[14] + x[10], 9);
|
||
x[6] ^= R(x[2] + x[14], 13);
|
||
x[10] ^= R(x[6] + x[2], 18);
|
||
x[3] ^= R(x[15] + x[11], 7);
|
||
x[7] ^= R(x[3] + x[15], 9);
|
||
x[11] ^= R(x[7] + x[3], 13);
|
||
x[15] ^= R(x[11] + x[7], 18);
|
||
x[1] ^= R(x[0] + x[3], 7);
|
||
x[2] ^= R(x[1] + x[0], 9);
|
||
x[3] ^= R(x[2] + x[1], 13);
|
||
x[0] ^= R(x[3] + x[2], 18);
|
||
x[6] ^= R(x[5] + x[4], 7);
|
||
x[7] ^= R(x[6] + x[5], 9);
|
||
x[4] ^= R(x[7] + x[6], 13);
|
||
x[5] ^= R(x[4] + x[7], 18);
|
||
x[11] ^= R(x[10] + x[9], 7);
|
||
x[8] ^= R(x[11] + x[10], 9);
|
||
x[9] ^= R(x[8] + x[11], 13);
|
||
x[10] ^= R(x[9] + x[8], 18);
|
||
x[12] ^= R(x[15] + x[14], 7);
|
||
x[13] ^= R(x[12] + x[15], 9);
|
||
x[14] ^= R(x[13] + x[12], 13);
|
||
x[15] ^= R(x[14] + x[13], 18);
|
||
}
|
||
|
||
for (i = 0; i < 16; ++i) B32[i] = x[i] + B32[i];
|
||
|
||
for (i = 0; i < 16; i++) {
|
||
var bi = i * 4;
|
||
B[bi + 0] = (B32[i] >> 0) & 0xff;
|
||
B[bi + 1] = (B32[i] >> 8) & 0xff;
|
||
B[bi + 2] = (B32[i] >> 16) & 0xff;
|
||
B[bi + 3] = (B32[i] >> 24) & 0xff;
|
||
}
|
||
}
|
||
|
||
function blockxor(S, Si, D, Di, len) {
|
||
var i = len >> 6;
|
||
while (i--) {
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
D[Di++] ^= S[Si++];
|
||
}
|
||
}
|
||
|
||
function integerify(B, bi, r) {
|
||
var n;
|
||
|
||
bi += (2 * r - 1) * 64;
|
||
|
||
n = (B[bi + 0] & 0xff) << 0;
|
||
n |= (B[bi + 1] & 0xff) << 8;
|
||
n |= (B[bi + 2] & 0xff) << 16;
|
||
n |= (B[bi + 3] & 0xff) << 24;
|
||
|
||
return n;
|
||
}
|
||
|
||
function arraycopy(src, srcPos, dest, destPos, length) {
|
||
while (length--) {
|
||
dest[destPos++] = src[srcPos++];
|
||
}
|
||
}
|
||
|
||
function arraycopy32(src, srcPos, dest, destPos, length) {
|
||
var i = length >> 5;
|
||
while (i--) {
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
dest[destPos++] = src[srcPos++];
|
||
}
|
||
}
|
||
} // scryptCore
|
||
}; // window.Crypto_scrypt
|
||
})();
|
||
|
||
/*!
|
||
* Crypto-JS v2.5.4 AES.js
|
||
* http://code.google.com/p/crypto-js/
|
||
* Copyright (c) 2009-2013, Jeff Mott. All rights reserved.
|
||
* http://code.google.com/p/crypto-js/wiki/License
|
||
*/
|
||
(function () {
|
||
// Shortcuts
|
||
var C = Crypto,
|
||
util = C.util,
|
||
charenc = C.charenc,
|
||
UTF8 = charenc.UTF8;
|
||
|
||
// Precomputed SBOX
|
||
var SBOX = [
|
||
0x63, 0x7c, 0x77, 0x7b, 0xf2, 0x6b, 0x6f, 0xc5, 0x30, 0x01, 0x67,
|
||
0x2b, 0xfe, 0xd7, 0xab, 0x76, 0xca, 0x82, 0xc9, 0x7d, 0xfa, 0x59,
|
||
0x47, 0xf0, 0xad, 0xd4, 0xa2, 0xaf, 0x9c, 0xa4, 0x72, 0xc0, 0xb7,
|
||
0xfd, 0x93, 0x26, 0x36, 0x3f, 0xf7, 0xcc, 0x34, 0xa5, 0xe5, 0xf1,
|
||
0x71, 0xd8, 0x31, 0x15, 0x04, 0xc7, 0x23, 0xc3, 0x18, 0x96, 0x05,
|
||
0x9a, 0x07, 0x12, 0x80, 0xe2, 0xeb, 0x27, 0xb2, 0x75, 0x09, 0x83,
|
||
0x2c, 0x1a, 0x1b, 0x6e, 0x5a, 0xa0, 0x52, 0x3b, 0xd6, 0xb3, 0x29,
|
||
0xe3, 0x2f, 0x84, 0x53, 0xd1, 0x00, 0xed, 0x20, 0xfc, 0xb1, 0x5b,
|
||
0x6a, 0xcb, 0xbe, 0x39, 0x4a, 0x4c, 0x58, 0xcf, 0xd0, 0xef, 0xaa,
|
||
0xfb, 0x43, 0x4d, 0x33, 0x85, 0x45, 0xf9, 0x02, 0x7f, 0x50, 0x3c,
|
||
0x9f, 0xa8, 0x51, 0xa3, 0x40, 0x8f, 0x92, 0x9d, 0x38, 0xf5, 0xbc,
|
||
0xb6, 0xda, 0x21, 0x10, 0xff, 0xf3, 0xd2, 0xcd, 0x0c, 0x13, 0xec,
|
||
0x5f, 0x97, 0x44, 0x17, 0xc4, 0xa7, 0x7e, 0x3d, 0x64, 0x5d, 0x19,
|
||
0x73, 0x60, 0x81, 0x4f, 0xdc, 0x22, 0x2a, 0x90, 0x88, 0x46, 0xee,
|
||
0xb8, 0x14, 0xde, 0x5e, 0x0b, 0xdb, 0xe0, 0x32, 0x3a, 0x0a, 0x49,
|
||
0x06, 0x24, 0x5c, 0xc2, 0xd3, 0xac, 0x62, 0x91, 0x95, 0xe4, 0x79,
|
||
0xe7, 0xc8, 0x37, 0x6d, 0x8d, 0xd5, 0x4e, 0xa9, 0x6c, 0x56, 0xf4,
|
||
0xea, 0x65, 0x7a, 0xae, 0x08, 0xba, 0x78, 0x25, 0x2e, 0x1c, 0xa6,
|
||
0xb4, 0xc6, 0xe8, 0xdd, 0x74, 0x1f, 0x4b, 0xbd, 0x8b, 0x8a, 0x70,
|
||
0x3e, 0xb5, 0x66, 0x48, 0x03, 0xf6, 0x0e, 0x61, 0x35, 0x57, 0xb9,
|
||
0x86, 0xc1, 0x1d, 0x9e, 0xe1, 0xf8, 0x98, 0x11, 0x69, 0xd9, 0x8e,
|
||
0x94, 0x9b, 0x1e, 0x87, 0xe9, 0xce, 0x55, 0x28, 0xdf, 0x8c, 0xa1,
|
||
0x89, 0x0d, 0xbf, 0xe6, 0x42, 0x68, 0x41, 0x99, 0x2d, 0x0f, 0xb0,
|
||
0x54, 0xbb, 0x16,
|
||
];
|
||
|
||
// Compute inverse SBOX lookup table
|
||
for (var INVSBOX = [], i = 0; i < 256; i++) INVSBOX[SBOX[i]] = i;
|
||
|
||
// Compute multiplication in GF(2^8) lookup tables
|
||
var MULT2 = [],
|
||
MULT3 = [],
|
||
MULT9 = [],
|
||
MULTB = [],
|
||
MULTD = [],
|
||
MULTE = [];
|
||
|
||
function xtime(a, b) {
|
||
for (var result = 0, i = 0; i < 8; i++) {
|
||
if (b & 1) result ^= a;
|
||
var hiBitSet = a & 0x80;
|
||
a = (a << 1) & 0xff;
|
||
if (hiBitSet) a ^= 0x1b;
|
||
b >>>= 1;
|
||
}
|
||
return result;
|
||
}
|
||
|
||
for (var i = 0; i < 256; i++) {
|
||
MULT2[i] = xtime(i, 2);
|
||
MULT3[i] = xtime(i, 3);
|
||
MULT9[i] = xtime(i, 9);
|
||
MULTB[i] = xtime(i, 0xb);
|
||
MULTD[i] = xtime(i, 0xd);
|
||
MULTE[i] = xtime(i, 0xe);
|
||
}
|
||
|
||
// Precomputed RCon lookup
|
||
var RCON = [
|
||
0x00, 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36,
|
||
];
|
||
|
||
// Inner state
|
||
var state = [[], [], [], []],
|
||
keylength,
|
||
nrounds,
|
||
keyschedule;
|
||
|
||
var AES = (C.AES = {
|
||
/**
|
||
* Public API
|
||
*/
|
||
|
||
encrypt: function (message, password, options) {
|
||
options = options || {};
|
||
|
||
// Determine mode
|
||
var mode = options.mode || new C.mode.OFB();
|
||
|
||
// Allow mode to override options
|
||
if (mode.fixOptions) mode.fixOptions(options);
|
||
|
||
var // Convert to bytes if message is a string
|
||
m =
|
||
message.constructor == String
|
||
? UTF8.stringToBytes(message)
|
||
: message,
|
||
// Generate random IV
|
||
iv = options.iv || util.randomBytes(AES._blocksize * 4),
|
||
// Generate key
|
||
k =
|
||
password.constructor == String
|
||
? // Derive key from pass-phrase
|
||
C.PBKDF2(password, iv, 32, {
|
||
asBytes: true,
|
||
})
|
||
: // else, assume byte array representing cryptographic key
|
||
password;
|
||
|
||
// Encrypt
|
||
AES._init(k);
|
||
mode.encrypt(AES, m, iv);
|
||
|
||
// Return ciphertext
|
||
m = options.iv ? m : iv.concat(m);
|
||
return options && options.asBytes ? m : util.bytesToBase64(m);
|
||
},
|
||
|
||
decrypt: function (ciphertext, password, options) {
|
||
options = options || {};
|
||
|
||
// Determine mode
|
||
var mode = options.mode || new C.mode.OFB();
|
||
|
||
// Allow mode to override options
|
||
if (mode.fixOptions) mode.fixOptions(options);
|
||
|
||
var // Convert to bytes if ciphertext is a string
|
||
c =
|
||
ciphertext.constructor == String
|
||
? util.base64ToBytes(ciphertext)
|
||
: ciphertext,
|
||
// Separate IV and message
|
||
iv = options.iv || c.splice(0, AES._blocksize * 4),
|
||
// Generate key
|
||
k =
|
||
password.constructor == String
|
||
? // Derive key from pass-phrase
|
||
C.PBKDF2(password, iv, 32, {
|
||
asBytes: true,
|
||
})
|
||
: // else, assume byte array representing cryptographic key
|
||
password;
|
||
|
||
// Decrypt
|
||
AES._init(k);
|
||
mode.decrypt(AES, c, iv);
|
||
|
||
// Return plaintext
|
||
return options && options.asBytes ? c : UTF8.bytesToString(c);
|
||
},
|
||
|
||
/**
|
||
* Package private methods and properties
|
||
*/
|
||
|
||
_blocksize: 4,
|
||
|
||
_encryptblock: function (m, offset) {
|
||
// Set input
|
||
for (var row = 0; row < AES._blocksize; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
state[row][col] = m[offset + col * 4 + row];
|
||
}
|
||
|
||
// Add round key
|
||
for (var row = 0; row < 4; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
state[row][col] ^= keyschedule[col][row];
|
||
}
|
||
|
||
for (var round = 1; round < nrounds; round++) {
|
||
// Sub bytes
|
||
for (var row = 0; row < 4; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
state[row][col] = SBOX[state[row][col]];
|
||
}
|
||
|
||
// Shift rows
|
||
state[1].push(state[1].shift());
|
||
state[2].push(state[2].shift());
|
||
state[2].push(state[2].shift());
|
||
state[3].unshift(state[3].pop());
|
||
|
||
// Mix columns
|
||
for (var col = 0; col < 4; col++) {
|
||
var s0 = state[0][col],
|
||
s1 = state[1][col],
|
||
s2 = state[2][col],
|
||
s3 = state[3][col];
|
||
|
||
state[0][col] = MULT2[s0] ^ MULT3[s1] ^ s2 ^ s3;
|
||
state[1][col] = s0 ^ MULT2[s1] ^ MULT3[s2] ^ s3;
|
||
state[2][col] = s0 ^ s1 ^ MULT2[s2] ^ MULT3[s3];
|
||
state[3][col] = MULT3[s0] ^ s1 ^ s2 ^ MULT2[s3];
|
||
}
|
||
|
||
// Add round key
|
||
for (var row = 0; row < 4; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
state[row][col] ^= keyschedule[round * 4 + col][row];
|
||
}
|
||
}
|
||
|
||
// Sub bytes
|
||
for (var row = 0; row < 4; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
state[row][col] = SBOX[state[row][col]];
|
||
}
|
||
|
||
// Shift rows
|
||
state[1].push(state[1].shift());
|
||
state[2].push(state[2].shift());
|
||
state[2].push(state[2].shift());
|
||
state[3].unshift(state[3].pop());
|
||
|
||
// Add round key
|
||
for (var row = 0; row < 4; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
state[row][col] ^= keyschedule[nrounds * 4 + col][row];
|
||
}
|
||
|
||
// Set output
|
||
for (var row = 0; row < AES._blocksize; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
m[offset + col * 4 + row] = state[row][col];
|
||
}
|
||
},
|
||
|
||
_decryptblock: function (c, offset) {
|
||
// Set input
|
||
for (var row = 0; row < AES._blocksize; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
state[row][col] = c[offset + col * 4 + row];
|
||
}
|
||
|
||
// Add round key
|
||
for (var row = 0; row < 4; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
state[row][col] ^= keyschedule[nrounds * 4 + col][row];
|
||
}
|
||
|
||
for (var round = 1; round < nrounds; round++) {
|
||
// Inv shift rows
|
||
state[1].unshift(state[1].pop());
|
||
state[2].push(state[2].shift());
|
||
state[2].push(state[2].shift());
|
||
state[3].push(state[3].shift());
|
||
|
||
// Inv sub bytes
|
||
for (var row = 0; row < 4; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
state[row][col] = INVSBOX[state[row][col]];
|
||
}
|
||
|
||
// Add round key
|
||
for (var row = 0; row < 4; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
state[row][col] ^=
|
||
keyschedule[(nrounds - round) * 4 + col][row];
|
||
}
|
||
|
||
// Inv mix columns
|
||
for (var col = 0; col < 4; col++) {
|
||
var s0 = state[0][col],
|
||
s1 = state[1][col],
|
||
s2 = state[2][col],
|
||
s3 = state[3][col];
|
||
|
||
state[0][col] = MULTE[s0] ^ MULTB[s1] ^ MULTD[s2] ^ MULT9[s3];
|
||
state[1][col] = MULT9[s0] ^ MULTE[s1] ^ MULTB[s2] ^ MULTD[s3];
|
||
state[2][col] = MULTD[s0] ^ MULT9[s1] ^ MULTE[s2] ^ MULTB[s3];
|
||
state[3][col] = MULTB[s0] ^ MULTD[s1] ^ MULT9[s2] ^ MULTE[s3];
|
||
}
|
||
}
|
||
|
||
// Inv shift rows
|
||
state[1].unshift(state[1].pop());
|
||
state[2].push(state[2].shift());
|
||
state[2].push(state[2].shift());
|
||
state[3].push(state[3].shift());
|
||
|
||
// Inv sub bytes
|
||
for (var row = 0; row < 4; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
state[row][col] = INVSBOX[state[row][col]];
|
||
}
|
||
|
||
// Add round key
|
||
for (var row = 0; row < 4; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
state[row][col] ^= keyschedule[col][row];
|
||
}
|
||
|
||
// Set output
|
||
for (var row = 0; row < AES._blocksize; row++) {
|
||
for (var col = 0; col < 4; col++)
|
||
c[offset + col * 4 + row] = state[row][col];
|
||
}
|
||
},
|
||
|
||
/**
|
||
* Private methods
|
||
*/
|
||
|
||
_init: function (k) {
|
||
keylength = k.length / 4;
|
||
nrounds = keylength + 6;
|
||
AES._keyexpansion(k);
|
||
},
|
||
|
||
// Generate a key schedule
|
||
_keyexpansion: function (k) {
|
||
keyschedule = [];
|
||
|
||
for (var row = 0; row < keylength; row++) {
|
||
keyschedule[row] = [
|
||
k[row * 4],
|
||
k[row * 4 + 1],
|
||
k[row * 4 + 2],
|
||
k[row * 4 + 3],
|
||
];
|
||
}
|
||
|
||
for (
|
||
var row = keylength;
|
||
row < AES._blocksize * (nrounds + 1);
|
||
row++
|
||
) {
|
||
var temp = [
|
||
keyschedule[row - 1][0],
|
||
keyschedule[row - 1][1],
|
||
keyschedule[row - 1][2],
|
||
keyschedule[row - 1][3],
|
||
];
|
||
|
||
if (row % keylength == 0) {
|
||
// Rot word
|
||
temp.push(temp.shift());
|
||
|
||
// Sub word
|
||
temp[0] = SBOX[temp[0]];
|
||
temp[1] = SBOX[temp[1]];
|
||
temp[2] = SBOX[temp[2]];
|
||
temp[3] = SBOX[temp[3]];
|
||
|
||
temp[0] ^= RCON[row / keylength];
|
||
} else if (keylength > 6 && row % keylength == 4) {
|
||
// Sub word
|
||
temp[0] = SBOX[temp[0]];
|
||
temp[1] = SBOX[temp[1]];
|
||
temp[2] = SBOX[temp[2]];
|
||
temp[3] = SBOX[temp[3]];
|
||
}
|
||
|
||
keyschedule[row] = [
|
||
keyschedule[row - keylength][0] ^ temp[0],
|
||
keyschedule[row - keylength][1] ^ temp[1],
|
||
keyschedule[row - keylength][2] ^ temp[2],
|
||
keyschedule[row - keylength][3] ^ temp[3],
|
||
];
|
||
}
|
||
},
|
||
});
|
||
})();
|
||
|
||
/*
|
||
* Copyright (c) 2000 - 2011 The Legion Of The Bouncy Castle (http://www.bouncycastle.org)
|
||
* Ported to JavaScript by bitaddress.org
|
||
*/
|
||
ec.FieldElementFp.fastLucasSequence = function (p, P, Q, k) {
|
||
// TODO Research and apply "common-multiplicand multiplication here"
|
||
|
||
var n = k.bitLength();
|
||
var s = k.getLowestSetBit();
|
||
var Uh = BigInteger.ONE;
|
||
var Vl = BigInteger.TWO;
|
||
var Vh = P;
|
||
var Ql = BigInteger.ONE;
|
||
var Qh = BigInteger.ONE;
|
||
|
||
for (var j = n - 1; j >= s + 1; --j) {
|
||
Ql = Ql.multiply(Qh).mod(p);
|
||
if (k.testBit(j)) {
|
||
Qh = Ql.multiply(Q).mod(p);
|
||
Uh = Uh.multiply(Vh).mod(p);
|
||
Vl = Vh.multiply(Vl).subtract(P.multiply(Ql)).mod(p);
|
||
Vh = Vh.multiply(Vh).subtract(Qh.shiftLeft(1)).mod(p);
|
||
} else {
|
||
Qh = Ql;
|
||
Uh = Uh.multiply(Vl).subtract(Ql).mod(p);
|
||
Vh = Vh.multiply(Vl).subtract(P.multiply(Ql)).mod(p);
|
||
Vl = Vl.multiply(Vl).subtract(Ql.shiftLeft(1)).mod(p);
|
||
}
|
||
}
|
||
|
||
Ql = Ql.multiply(Qh).mod(p);
|
||
Qh = Ql.multiply(Q).mod(p);
|
||
Uh = Uh.multiply(Vl).subtract(Ql).mod(p);
|
||
Vl = Vh.multiply(Vl).subtract(P.multiply(Ql)).mod(p);
|
||
Ql = Ql.multiply(Qh).mod(p);
|
||
|
||
for (var j = 1; j <= s; ++j) {
|
||
Uh = Uh.multiply(Vl).mod(p);
|
||
Vl = Vl.multiply(Vl).subtract(Ql.shiftLeft(1)).mod(p);
|
||
Ql = Ql.multiply(Ql).mod(p);
|
||
}
|
||
|
||
return [Uh, Vl];
|
||
};
|
||
|
||
// ----------------
|
||
// ECPointFp constructor
|
||
ec.PointFp = function (curve, x, y, z, compressed) {
|
||
this.curve = curve;
|
||
this.x = x;
|
||
this.y = y;
|
||
// Projective coordinates: either zinv == null or z * zinv == 1
|
||
// z and zinv are just BigIntegers, not fieldElements
|
||
if (z == null) {
|
||
this.z = BigInteger.ONE;
|
||
} else {
|
||
this.z = z;
|
||
}
|
||
this.zinv = null;
|
||
// compression flag
|
||
this.compressed = !!compressed;
|
||
};
|
||
|
||
ec.PointFp.prototype.getX = function () {
|
||
if (this.zinv == null) {
|
||
this.zinv = this.z.modInverse(this.curve.q);
|
||
}
|
||
var r = this.x.toBigInteger().multiply(this.zinv);
|
||
this.curve.reduce(r);
|
||
return this.curve.fromBigInteger(r);
|
||
};
|
||
|
||
ec.PointFp.prototype.getY = function () {
|
||
if (this.zinv == null) {
|
||
this.zinv = this.z.modInverse(this.curve.q);
|
||
}
|
||
var r = this.y.toBigInteger().multiply(this.zinv);
|
||
this.curve.reduce(r);
|
||
return this.curve.fromBigInteger(r);
|
||
};
|
||
|
||
ec.PointFp.prototype.equals = function (other) {
|
||
if (other == this) return true;
|
||
if (this.isInfinity()) return other.isInfinity();
|
||
if (other.isInfinity()) return this.isInfinity();
|
||
var u, v;
|
||
// u = Y2 * Z1 - Y1 * Z2
|
||
u = other.y
|
||
.toBigInteger()
|
||
.multiply(this.z)
|
||
.subtract(this.y.toBigInteger().multiply(other.z))
|
||
.mod(this.curve.q);
|
||
if (!u.equals(BigInteger.ZERO)) return false;
|
||
// v = X2 * Z1 - X1 * Z2
|
||
v = other.x
|
||
.toBigInteger()
|
||
.multiply(this.z)
|
||
.subtract(this.x.toBigInteger().multiply(other.z))
|
||
.mod(this.curve.q);
|
||
return v.equals(BigInteger.ZERO);
|
||
};
|
||
|
||
ec.PointFp.prototype.isInfinity = function () {
|
||
if (this.x == null && this.y == null) return true;
|
||
return (
|
||
this.z.equals(BigInteger.ZERO) &&
|
||
!this.y.toBigInteger().equals(BigInteger.ZERO)
|
||
);
|
||
};
|
||
|
||
ec.PointFp.prototype.negate = function () {
|
||
return new ec.PointFp(this.curve, this.x, this.y.negate(), this.z);
|
||
};
|
||
|
||
ec.PointFp.prototype.add = function (b) {
|
||
if (this.isInfinity()) return b;
|
||
if (b.isInfinity()) return this;
|
||
|
||
// u = Y2 * Z1 - Y1 * Z2
|
||
var u = b.y
|
||
.toBigInteger()
|
||
.multiply(this.z)
|
||
.subtract(this.y.toBigInteger().multiply(b.z))
|
||
.mod(this.curve.q);
|
||
// v = X2 * Z1 - X1 * Z2
|
||
var v = b.x
|
||
.toBigInteger()
|
||
.multiply(this.z)
|
||
.subtract(this.x.toBigInteger().multiply(b.z))
|
||
.mod(this.curve.q);
|
||
|
||
if (BigInteger.ZERO.equals(v)) {
|
||
if (BigInteger.ZERO.equals(u)) {
|
||
return this.twice(); // this == b, so double
|
||
}
|
||
return this.curve.getInfinity(); // this = -b, so infinity
|
||
}
|
||
|
||
var THREE = new BigInteger("3");
|
||
var x1 = this.x.toBigInteger();
|
||
var y1 = this.y.toBigInteger();
|
||
var x2 = b.x.toBigInteger();
|
||
var y2 = b.y.toBigInteger();
|
||
|
||
var v2 = v.square();
|
||
var v3 = v2.multiply(v);
|
||
var x1v2 = x1.multiply(v2);
|
||
var zu2 = u.square().multiply(this.z);
|
||
|
||
// x3 = v * (z2 * (z1 * u^2 - 2 * x1 * v^2) - v^3)
|
||
var x3 = zu2
|
||
.subtract(x1v2.shiftLeft(1))
|
||
.multiply(b.z)
|
||
.subtract(v3)
|
||
.multiply(v)
|
||
.mod(this.curve.q);
|
||
// y3 = z2 * (3 * x1 * u * v^2 - y1 * v^3 - z1 * u^3) + u * v^3
|
||
var y3 = x1v2
|
||
.multiply(THREE)
|
||
.multiply(u)
|
||
.subtract(y1.multiply(v3))
|
||
.subtract(zu2.multiply(u))
|
||
.multiply(b.z)
|
||
.add(u.multiply(v3))
|
||
.mod(this.curve.q);
|
||
// z3 = v^3 * z1 * z2
|
||
var z3 = v3.multiply(this.z).multiply(b.z).mod(this.curve.q);
|
||
|
||
return new ec.PointFp(
|
||
this.curve,
|
||
this.curve.fromBigInteger(x3),
|
||
this.curve.fromBigInteger(y3),
|
||
z3
|
||
);
|
||
};
|
||
|
||
ec.PointFp.prototype.twice = function () {
|
||
if (this.isInfinity()) return this;
|
||
if (this.y.toBigInteger().signum() == 0)
|
||
return this.curve.getInfinity();
|
||
|
||
// TODO: optimized handling of constants
|
||
var THREE = new BigInteger("3");
|
||
var x1 = this.x.toBigInteger();
|
||
var y1 = this.y.toBigInteger();
|
||
|
||
var y1z1 = y1.multiply(this.z);
|
||
var y1sqz1 = y1z1.multiply(y1).mod(this.curve.q);
|
||
var a = this.curve.a.toBigInteger();
|
||
|
||
// w = 3 * x1^2 + a * z1^2
|
||
var w = x1.square().multiply(THREE);
|
||
if (!BigInteger.ZERO.equals(a)) {
|
||
w = w.add(this.z.square().multiply(a));
|
||
}
|
||
w = w.mod(this.curve.q);
|
||
//this.curve.reduce(w);
|
||
// x3 = 2 * y1 * z1 * (w^2 - 8 * x1 * y1^2 * z1)
|
||
var x3 = w
|
||
.square()
|
||
.subtract(x1.shiftLeft(3).multiply(y1sqz1))
|
||
.shiftLeft(1)
|
||
.multiply(y1z1)
|
||
.mod(this.curve.q);
|
||
// y3 = 4 * y1^2 * z1 * (3 * w * x1 - 2 * y1^2 * z1) - w^3
|
||
var y3 = w
|
||
.multiply(THREE)
|
||
.multiply(x1)
|
||
.subtract(y1sqz1.shiftLeft(1))
|
||
.shiftLeft(2)
|
||
.multiply(y1sqz1)
|
||
.subtract(w.square().multiply(w))
|
||
.mod(this.curve.q);
|
||
// z3 = 8 * (y1 * z1)^3
|
||
var z3 = y1z1.square().multiply(y1z1).shiftLeft(3).mod(this.curve.q);
|
||
|
||
return new ec.PointFp(
|
||
this.curve,
|
||
this.curve.fromBigInteger(x3),
|
||
this.curve.fromBigInteger(y3),
|
||
z3
|
||
);
|
||
};
|
||
|
||
// Simple NAF (Non-Adjacent Form) multiplication algorithm
|
||
// TODO: modularize the multiplication algorithm
|
||
ec.PointFp.prototype.multiply = function (k) {
|
||
if (this.isInfinity()) return this;
|
||
if (k.signum() == 0) return this.curve.getInfinity();
|
||
|
||
var e = k;
|
||
var h = e.multiply(new BigInteger("3"));
|
||
|
||
var neg = this.negate();
|
||
var R = this;
|
||
|
||
var i;
|
||
for (i = h.bitLength() - 2; i > 0; --i) {
|
||
R = R.twice();
|
||
|
||
var hBit = h.testBit(i);
|
||
var eBit = e.testBit(i);
|
||
|
||
if (hBit != eBit) {
|
||
R = R.add(hBit ? this : neg);
|
||
}
|
||
}
|
||
|
||
return R;
|
||
};
|
||
|
||
// Compute this*j + x*k (simultaneous multiplication)
|
||
ec.PointFp.prototype.multiplyTwo = function (j, x, k) {
|
||
var i;
|
||
if (j.bitLength() > k.bitLength()) i = j.bitLength() - 1;
|
||
else i = k.bitLength() - 1;
|
||
|
||
var R = this.curve.getInfinity();
|
||
var both = this.add(x);
|
||
while (i >= 0) {
|
||
R = R.twice();
|
||
if (j.testBit(i)) {
|
||
if (k.testBit(i)) {
|
||
R = R.add(both);
|
||
} else {
|
||
R = R.add(this);
|
||
}
|
||
} else {
|
||
if (k.testBit(i)) {
|
||
R = R.add(x);
|
||
}
|
||
}
|
||
--i;
|
||
}
|
||
|
||
return R;
|
||
};
|
||
|
||
// patched by bitaddress.org and Casascius for use with Bitcoin.ECKey
|
||
// patched by coretechs to support compressed public keys
|
||
ec.PointFp.prototype.getEncoded = function (compressed) {
|
||
var x = this.getX().toBigInteger();
|
||
var y = this.getY().toBigInteger();
|
||
var len = 32; // integerToBytes will zero pad if integer is less than 32 bytes. 32 bytes length is required by the Bitcoin protocol.
|
||
var enc = ec.integerToBytes(x, len);
|
||
|
||
// when compressed prepend byte depending if y point is even or odd
|
||
if (compressed) {
|
||
if (y.isEven()) {
|
||
enc.unshift(0x02);
|
||
} else {
|
||
enc.unshift(0x03);
|
||
}
|
||
} else {
|
||
enc.unshift(0x04);
|
||
enc = enc.concat(ec.integerToBytes(y, len)); // uncompressed public key appends the bytes of the y point
|
||
}
|
||
return enc;
|
||
};
|
||
|
||
ec.PointFp.decodeFrom = function (curve, enc) {
|
||
var type = enc[0];
|
||
var dataLen = enc.length - 1;
|
||
|
||
// Extract x and y as byte arrays
|
||
var xBa = enc.slice(1, 1 + dataLen / 2);
|
||
var yBa = enc.slice(1 + dataLen / 2, 1 + dataLen);
|
||
|
||
// Prepend zero byte to prevent interpretation as negative integer
|
||
xBa.unshift(0);
|
||
yBa.unshift(0);
|
||
|
||
// Convert to BigIntegers
|
||
var x = new BigInteger(xBa);
|
||
var y = new BigInteger(yBa);
|
||
|
||
// Return point
|
||
return new ec.PointFp(
|
||
curve,
|
||
curve.fromBigInteger(x),
|
||
curve.fromBigInteger(y)
|
||
);
|
||
};
|
||
|
||
ec.PointFp.prototype.add2D = function (b) {
|
||
if (this.isInfinity()) return b;
|
||
if (b.isInfinity()) return this;
|
||
|
||
if (this.x.equals(b.x)) {
|
||
if (this.y.equals(b.y)) {
|
||
// this = b, i.e. this must be doubled
|
||
return this.twice();
|
||
}
|
||
// this = -b, i.e. the result is the point at infinity
|
||
return this.curve.getInfinity();
|
||
}
|
||
|
||
var x_x = b.x.subtract(this.x);
|
||
var y_y = b.y.subtract(this.y);
|
||
var gamma = y_y.divide(x_x);
|
||
|
||
var x3 = gamma.square().subtract(this.x).subtract(b.x);
|
||
var y3 = gamma.multiply(this.x.subtract(x3)).subtract(this.y);
|
||
|
||
return new ec.PointFp(this.curve, x3, y3);
|
||
};
|
||
|
||
ec.PointFp.prototype.twice2D = function () {
|
||
if (this.isInfinity()) return this;
|
||
if (this.y.toBigInteger().signum() == 0) {
|
||
// if y1 == 0, then (x1, y1) == (x1, -y1)
|
||
// and hence this = -this and thus 2(x1, y1) == infinity
|
||
return this.curve.getInfinity();
|
||
}
|
||
|
||
var TWO = this.curve.fromBigInteger(BigInteger.valueOf(2));
|
||
var THREE = this.curve.fromBigInteger(BigInteger.valueOf(3));
|
||
var gamma = this.x
|
||
.square()
|
||
.multiply(THREE)
|
||
.add(this.curve.a)
|
||
.divide(this.y.multiply(TWO));
|
||
|
||
var x3 = gamma.square().subtract(this.x.multiply(TWO));
|
||
var y3 = gamma.multiply(this.x.subtract(x3)).subtract(this.y);
|
||
|
||
return new ec.PointFp(this.curve, x3, y3);
|
||
};
|
||
|
||
ec.PointFp.prototype.multiply2D = function (k) {
|
||
if (this.isInfinity()) return this;
|
||
if (k.signum() == 0) return this.curve.getInfinity();
|
||
|
||
var e = k;
|
||
var h = e.multiply(new BigInteger("3"));
|
||
|
||
var neg = this.negate();
|
||
var R = this;
|
||
|
||
var i;
|
||
for (i = h.bitLength() - 2; i > 0; --i) {
|
||
R = R.twice();
|
||
|
||
var hBit = h.testBit(i);
|
||
var eBit = e.testBit(i);
|
||
|
||
if (hBit != eBit) {
|
||
R = R.add2D(hBit ? this : neg);
|
||
}
|
||
}
|
||
|
||
return R;
|
||
};
|
||
|
||
ec.PointFp.prototype.isOnCurve = function () {
|
||
var x = this.getX().toBigInteger();
|
||
var y = this.getY().toBigInteger();
|
||
var a = this.curve.getA().toBigInteger();
|
||
var b = this.curve.getB().toBigInteger();
|
||
var n = this.curve.getQ();
|
||
var lhs = y.multiply(y).mod(n);
|
||
var rhs = x.multiply(x).multiply(x).add(a.multiply(x)).add(b).mod(n);
|
||
return lhs.equals(rhs);
|
||
};
|
||
|
||
ec.PointFp.prototype.toString = function () {
|
||
return (
|
||
"(" +
|
||
this.getX().toBigInteger().toString() +
|
||
"," +
|
||
this.getY().toBigInteger().toString() +
|
||
")"
|
||
);
|
||
};
|
||
|
||
/**
|
||
* Validate an elliptic curve point.
|
||
*
|
||
* See SEC 1, section 3.2.2.1: Elliptic Curve Public Key Validation Primitive
|
||
*/
|
||
ec.PointFp.prototype.validate = function () {
|
||
var n = this.curve.getQ();
|
||
|
||
// Check Q != O
|
||
if (this.isInfinity()) {
|
||
throw new Error("Point is at infinity.");
|
||
}
|
||
|
||
// Check coordinate bounds
|
||
var x = this.getX().toBigInteger();
|
||
var y = this.getY().toBigInteger();
|
||
if (
|
||
x.compareTo(BigInteger.ONE) < 0 ||
|
||
x.compareTo(n.subtract(BigInteger.ONE)) > 0
|
||
) {
|
||
throw new Error("x coordinate out of bounds");
|
||
}
|
||
if (
|
||
y.compareTo(BigInteger.ONE) < 0 ||
|
||
y.compareTo(n.subtract(BigInteger.ONE)) > 0
|
||
) {
|
||
throw new Error("y coordinate out of bounds");
|
||
}
|
||
|
||
// Check y^2 = x^3 + ax + b (mod n)
|
||
if (!this.isOnCurve()) {
|
||
throw new Error("Point is not on the curve.");
|
||
}
|
||
|
||
// Check nQ = 0 (Q is a scalar multiple of G)
|
||
if (this.multiply(n).isInfinity()) {
|
||
// TODO: This check doesn't work - fix.
|
||
throw new Error("Point is not a scalar multiple of G.");
|
||
}
|
||
|
||
return true;
|
||
};
|
||
|
||
// ----------------
|
||
// ECCurveFp constructor
|
||
ec.CurveFp = function (q, a, b) {
|
||
this.q = q;
|
||
this.a = this.fromBigInteger(a);
|
||
this.b = this.fromBigInteger(b);
|
||
this.infinity = new ec.PointFp(this, null, null);
|
||
this.reducer = new Barrett(this.q);
|
||
};
|
||
|
||
ec.CurveFp.prototype.getQ = function () {
|
||
return this.q;
|
||
};
|
||
|
||
ec.CurveFp.prototype.getA = function () {
|
||
return this.a;
|
||
};
|
||
|
||
ec.CurveFp.prototype.getB = function () {
|
||
return this.b;
|
||
};
|
||
|
||
ec.CurveFp.prototype.equals = function (other) {
|
||
if (other == this) return true;
|
||
return (
|
||
this.q.equals(other.q) &&
|
||
this.a.equals(other.a) &&
|
||
this.b.equals(other.b)
|
||
);
|
||
};
|
||
|
||
ec.CurveFp.prototype.getInfinity = function () {
|
||
return this.infinity;
|
||
};
|
||
|
||
ec.CurveFp.prototype.fromBigInteger = function (x) {
|
||
return new ec.FieldElementFp(this.q, x);
|
||
};
|
||
|
||
ec.CurveFp.prototype.reduce = function (x) {
|
||
this.reducer.reduce(x);
|
||
};
|
||
|
||
// for now, work with hex strings because they're easier in JS
|
||
// compressed support added by bitaddress.org
|
||
ec.CurveFp.prototype.decodePointHex = function (s) {
|
||
var firstByte = parseInt(s.substr(0, 2), 16);
|
||
switch (
|
||
firstByte // first byte
|
||
) {
|
||
case 0:
|
||
return this.infinity;
|
||
case 2: // compressed
|
||
case 3: // compressed
|
||
var yTilde = firstByte & 1;
|
||
var xHex = s.substr(2, s.length - 2);
|
||
var X1 = new BigInteger(xHex, 16);
|
||
return this.decompressPoint(yTilde, X1);
|
||
case 4: // uncompressed
|
||
case 6: // hybrid
|
||
case 7: // hybrid
|
||
var len = (s.length - 2) / 2;
|
||
var xHex = s.substr(2, len);
|
||
var yHex = s.substr(len + 2, len);
|
||
|
||
return new ec.PointFp(
|
||
this,
|
||
this.fromBigInteger(new BigInteger(xHex, 16)),
|
||
this.fromBigInteger(new BigInteger(yHex, 16))
|
||
);
|
||
|
||
default:
|
||
// unsupported
|
||
return null;
|
||
}
|
||
};
|
||
|
||
ec.CurveFp.prototype.encodePointHex = function (p) {
|
||
if (p.isInfinity()) return "00";
|
||
var xHex = p.getX().toBigInteger().toString(16);
|
||
var yHex = p.getY().toBigInteger().toString(16);
|
||
var oLen = this.getQ().toString(16).length;
|
||
if (oLen % 2 != 0) oLen++;
|
||
while (xHex.length < oLen) {
|
||
xHex = "0" + xHex;
|
||
}
|
||
while (yHex.length < oLen) {
|
||
yHex = "0" + yHex;
|
||
}
|
||
return "04" + xHex + yHex;
|
||
};
|
||
|
||
/*
|
||
* Copyright (c) 2000 - 2011 The Legion Of The Bouncy Castle (http://www.bouncycastle.org)
|
||
* Ported to JavaScript by bitaddress.org
|
||
*
|
||
* Number yTilde
|
||
* BigInteger X1
|
||
*/
|
||
ec.CurveFp.prototype.decompressPoint = function (yTilde, X1) {
|
||
var x = this.fromBigInteger(X1);
|
||
var alpha = x.multiply(x.square().add(this.getA())).add(this.getB());
|
||
var beta = alpha.sqrt();
|
||
// if we can't find a sqrt we haven't got a point on the curve - run!
|
||
if (beta == null) throw new Error("Invalid point compression");
|
||
var betaValue = beta.toBigInteger();
|
||
var bit0 = betaValue.testBit(0) ? 1 : 0;
|
||
if (bit0 != yTilde) {
|
||
// Use the other root
|
||
beta = this.fromBigInteger(this.getQ().subtract(betaValue));
|
||
}
|
||
return new ec.PointFp(this, x, beta, null, true);
|
||
};
|
||
|
||
ec.fromHex = function (s) {
|
||
return new BigInteger(s, 16);
|
||
};
|
||
|
||
ec.integerToBytes = function (i, len) {
|
||
var bytes = i.toByteArrayUnsigned();
|
||
if (len < bytes.length) {
|
||
bytes = bytes.slice(bytes.length - len);
|
||
} else
|
||
while (len > bytes.length) {
|
||
bytes.unshift(0);
|
||
}
|
||
return bytes;
|
||
};
|
||
|
||
// Named EC curves
|
||
// ----------------
|
||
// X9ECParameters constructor
|
||
ec.X9Parameters = function (curve, g, n, h) {
|
||
this.curve = curve;
|
||
this.g = g;
|
||
this.n = n;
|
||
this.h = h;
|
||
};
|
||
ec.X9Parameters.prototype.getCurve = function () {
|
||
return this.curve;
|
||
};
|
||
ec.X9Parameters.prototype.getG = function () {
|
||
return this.g;
|
||
};
|
||
ec.X9Parameters.prototype.getN = function () {
|
||
return this.n;
|
||
};
|
||
ec.X9Parameters.prototype.getH = function () {
|
||
return this.h;
|
||
};
|
||
|
||
// secp256k1 is the Curve used by Bitcoin
|
||
ec.secNamedCurves = {
|
||
// used by Bitcoin
|
||
secp256k1: function () {
|
||
// p = 2^256 - 2^32 - 2^9 - 2^8 - 2^7 - 2^6 - 2^4 - 1
|
||
var p = ec.fromHex(
|
||
"FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFC2F"
|
||
);
|
||
var a = BigInteger.ZERO;
|
||
var b = ec.fromHex("7");
|
||
var n = ec.fromHex(
|
||
"FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141"
|
||
);
|
||
var h = BigInteger.ONE;
|
||
var curve = new ec.CurveFp(p, a, b);
|
||
var G = curve.decodePointHex(
|
||
"04" +
|
||
"79BE667EF9DCBBAC55A06295CE870B07029BFCDB2DCE28D959F2815B16F81798" +
|
||
"483ADA7726A3C4655DA4FBFC0E1108A8FD17B448A68554199C47D08FFB10D4B8"
|
||
);
|
||
return new ec.X9Parameters(curve, G, n, h);
|
||
},
|
||
};
|
||
|
||
// secp256k1 called by Bitcoin's ECKEY
|
||
ec.getSECCurveByName = function (name) {
|
||
if (ec.secNamedCurves[name] == undefined) return null;
|
||
return ec.secNamedCurves[name]();
|
||
};
|
||
})();
|
||
|
||
//bitTrx.js
|
||
(function () {
|
||
var bitjs = (window.bitjs = function () {});
|
||
|
||
/* public vars */
|
||
bitjs.pub = 0x23; // flochange - changed the prefix to FLO Mainnet PublicKey Prefix 0x23
|
||
bitjs.priv = 0xa3; //flochange - changed the prefix to FLO Mainnet Private key prefix 0xa3
|
||
bitjs.compressed = false;
|
||
|
||
/* provide a privkey and return an WIF */
|
||
bitjs.privkey2wif = function (h) {
|
||
var r = Crypto.util.hexToBytes(h);
|
||
|
||
if (bitjs.compressed == true) {
|
||
r.push(0x01);
|
||
}
|
||
|
||
r.unshift(bitjs.priv);
|
||
var hash = Crypto.SHA256(
|
||
Crypto.SHA256(r, {
|
||
asBytes: true,
|
||
}),
|
||
{
|
||
asBytes: true,
|
||
}
|
||
);
|
||
var checksum = hash.slice(0, 4);
|
||
|
||
return B58.encode(r.concat(checksum));
|
||
};
|
||
|
||
/* convert a wif key back to a private key */
|
||
bitjs.wif2privkey = function (wif) {
|
||
var compressed = false;
|
||
var decode = B58.decode(wif);
|
||
var key = decode.slice(0, decode.length - 4);
|
||
key = key.slice(1, key.length);
|
||
if (key.length >= 33 && key[key.length - 1] == 0x01) {
|
||
key = key.slice(0, key.length - 1);
|
||
compressed = true;
|
||
}
|
||
return {
|
||
privkey: Crypto.util.bytesToHex(key),
|
||
compressed: compressed,
|
||
};
|
||
};
|
||
|
||
/* convert a wif to a pubkey */
|
||
bitjs.wif2pubkey = function (wif) {
|
||
var compressed = bitjs.compressed;
|
||
var r = bitjs.wif2privkey(wif);
|
||
bitjs.compressed = r["compressed"];
|
||
var pubkey = bitjs.newPubkey(r["privkey"]);
|
||
bitjs.compressed = compressed;
|
||
return {
|
||
pubkey: pubkey,
|
||
compressed: r["compressed"],
|
||
};
|
||
};
|
||
|
||
/* convert a wif to a address */
|
||
bitjs.wif2address = function (wif) {
|
||
var r = bitjs.wif2pubkey(wif);
|
||
return {
|
||
address: bitjs.pubkey2address(r["pubkey"]),
|
||
compressed: r["compressed"],
|
||
};
|
||
};
|
||
|
||
/* generate a public key from a private key */
|
||
bitjs.newPubkey = function (hash) {
|
||
var privateKeyBigInt = BigInteger.fromByteArrayUnsigned(
|
||
Crypto.util.hexToBytes(hash)
|
||
);
|
||
var curve = EllipticCurve.getSECCurveByName("secp256k1");
|
||
|
||
var curvePt = curve.getG().multiply(privateKeyBigInt);
|
||
var x = curvePt.getX().toBigInteger();
|
||
var y = curvePt.getY().toBigInteger();
|
||
|
||
var publicKeyBytes = EllipticCurve.integerToBytes(x, 32);
|
||
publicKeyBytes = publicKeyBytes.concat(
|
||
EllipticCurve.integerToBytes(y, 32)
|
||
);
|
||
publicKeyBytes.unshift(0x04);
|
||
|
||
if (bitjs.compressed == true) {
|
||
var publicKeyBytesCompressed = EllipticCurve.integerToBytes(x, 32);
|
||
if (y.isEven()) {
|
||
publicKeyBytesCompressed.unshift(0x02);
|
||
} else {
|
||
publicKeyBytesCompressed.unshift(0x03);
|
||
}
|
||
return Crypto.util.bytesToHex(publicKeyBytesCompressed);
|
||
} else {
|
||
return Crypto.util.bytesToHex(publicKeyBytes);
|
||
}
|
||
};
|
||
|
||
/* provide a public key and return address */
|
||
bitjs.pubkey2address = function (h, byte) {
|
||
var r = ripemd160(
|
||
Crypto.SHA256(Crypto.util.hexToBytes(h), {
|
||
asBytes: true,
|
||
})
|
||
);
|
||
r.unshift(byte || bitjs.pub);
|
||
var hash = Crypto.SHA256(
|
||
Crypto.SHA256(r, {
|
||
asBytes: true,
|
||
}),
|
||
{
|
||
asBytes: true,
|
||
}
|
||
);
|
||
var checksum = hash.slice(0, 4);
|
||
return B58.encode(r.concat(checksum));
|
||
};
|
||
|
||
bitjs.transaction = function () {
|
||
var btrx = {};
|
||
btrx.version = 2; //flochange look at this version
|
||
btrx.inputs = [];
|
||
btrx.outputs = [];
|
||
btrx.locktime = 0;
|
||
btrx.floData = ""; //flochange .. look at this
|
||
|
||
btrx.addinput = function (txid, index, scriptPubKey, sequence) {
|
||
var o = {};
|
||
o.outpoint = {
|
||
hash: txid,
|
||
index: index,
|
||
};
|
||
//o.script = []; Signature and Public Key should be added after singning
|
||
o.script = Crypto.util.hexToBytes(scriptPubKey); //push previous output pubkey script
|
||
o.sequence = sequence || (btrx.locktime == 0 ? 4294967295 : 0);
|
||
return this.inputs.push(o);
|
||
};
|
||
|
||
btrx.addoutput = function (address, value) {
|
||
var o = {};
|
||
var buf = [];
|
||
var addrDecoded = btrx.addressDecode(address);
|
||
o.value = new BigInteger("" + Math.round(value * 1 * 1e8), 10);
|
||
buf.push(118); //OP_DUP
|
||
buf.push(169); //OP_HASH160
|
||
buf.push(addrDecoded.length);
|
||
buf = buf.concat(addrDecoded); // address in bytes
|
||
buf.push(136); //OP_EQUALVERIFY
|
||
buf.push(172); // OP_CHECKSIG
|
||
o.script = buf;
|
||
return this.outputs.push(o);
|
||
};
|
||
|
||
btrx.addflodata = function (txcomments) {
|
||
// flochange - this whole function needs to be done
|
||
this.floData = txcomments;
|
||
return this.floData; //flochange .. returning the txcomments -- check if the function return will assign
|
||
};
|
||
|
||
// Only standard addresses
|
||
btrx.addressDecode = function (address) {
|
||
var bytes = B58.decode(address);
|
||
var front = bytes.slice(0, bytes.length - 4);
|
||
var back = bytes.slice(bytes.length - 4);
|
||
var checksum = Crypto.SHA256(
|
||
Crypto.SHA256(front, {
|
||
asBytes: true,
|
||
}),
|
||
{
|
||
asBytes: true,
|
||
}
|
||
).slice(0, 4);
|
||
if (checksum + "" == back + "") {
|
||
return front.slice(1);
|
||
}
|
||
};
|
||
|
||
/* generate the transaction hash to sign from a transaction input */
|
||
btrx.transactionHash = function (index, sigHashType) {
|
||
var clone = bitjs.clone(this);
|
||
var shType = sigHashType || 1;
|
||
|
||
/* black out all other ins, except this one */
|
||
for (var i = 0; i < clone.inputs.length; i++) {
|
||
if (index != i) {
|
||
clone.inputs[i].script = [];
|
||
}
|
||
}
|
||
|
||
if (clone.inputs && clone.inputs[index]) {
|
||
/* SIGHASH : For more info on sig hashs see https://en.bitcoin.it/wiki/OP_CHECKSIG
|
||
and https://bitcoin.org/en/developer-guide#signature-hash-type */
|
||
|
||
if (shType == 1) {
|
||
//SIGHASH_ALL 0x01
|
||
} else if (shType == 2) {
|
||
//SIGHASH_NONE 0x02
|
||
clone.outputs = [];
|
||
for (var i = 0; i < clone.inputs.length; i++) {
|
||
if (index != i) {
|
||
clone.inputs[i].sequence = 0;
|
||
}
|
||
}
|
||
} else if (shType == 3) {
|
||
//SIGHASH_SINGLE 0x03
|
||
clone.outputs.length = index + 1;
|
||
|
||
for (var i = 0; i < index; i++) {
|
||
clone.outputs[i].value = -1;
|
||
clone.outputs[i].script = [];
|
||
}
|
||
|
||
for (var i = 0; i < clone.inputs.length; i++) {
|
||
if (index != i) {
|
||
clone.inputs[i].sequence = 0;
|
||
}
|
||
}
|
||
} else if (shType >= 128) {
|
||
//SIGHASH_ANYONECANPAY 0x80
|
||
clone.inputs = [clone.inputs[index]];
|
||
|
||
if (shType == 129) {
|
||
// SIGHASH_ALL + SIGHASH_ANYONECANPAY
|
||
} else if (shType == 130) {
|
||
// SIGHASH_NONE + SIGHASH_ANYONECANPAY
|
||
clone.outputs = [];
|
||
} else if (shType == 131) {
|
||
// SIGHASH_SINGLE + SIGHASH_ANYONECANPAY
|
||
clone.outputs.length = index + 1;
|
||
for (var i = 0; i < index; i++) {
|
||
clone.outputs[i].value = -1;
|
||
clone.outputs[i].script = [];
|
||
}
|
||
}
|
||
}
|
||
|
||
var buffer = Crypto.util.hexToBytes(clone.serialize());
|
||
buffer = buffer.concat(bitjs.numToBytes(parseInt(shType), 4));
|
||
var hash = Crypto.SHA256(buffer, {
|
||
asBytes: true,
|
||
});
|
||
var r = Crypto.util.bytesToHex(
|
||
Crypto.SHA256(hash, {
|
||
asBytes: true,
|
||
})
|
||
);
|
||
return r;
|
||
} else {
|
||
return false;
|
||
}
|
||
};
|
||
|
||
/* generate a signature from a transaction hash */
|
||
btrx.transactionSig = function (index, wif, sigHashType, txhash) {
|
||
function serializeSig(r, s) {
|
||
var rBa = r.toByteArraySigned();
|
||
var sBa = s.toByteArraySigned();
|
||
|
||
var sequence = [];
|
||
sequence.push(0x02); // INTEGER
|
||
sequence.push(rBa.length);
|
||
sequence = sequence.concat(rBa);
|
||
|
||
sequence.push(0x02); // INTEGER
|
||
sequence.push(sBa.length);
|
||
sequence = sequence.concat(sBa);
|
||
|
||
sequence.unshift(sequence.length);
|
||
sequence.unshift(0x30); // SEQUENCE
|
||
|
||
return sequence;
|
||
}
|
||
|
||
var shType = sigHashType || 1;
|
||
var hash =
|
||
txhash ||
|
||
Crypto.util.hexToBytes(this.transactionHash(index, shType));
|
||
|
||
if (hash) {
|
||
var curve = EllipticCurve.getSECCurveByName("secp256k1");
|
||
var key = bitjs.wif2privkey(wif);
|
||
var priv = BigInteger.fromByteArrayUnsigned(
|
||
Crypto.util.hexToBytes(key["privkey"])
|
||
);
|
||
var n = curve.getN();
|
||
var e = BigInteger.fromByteArrayUnsigned(hash);
|
||
var badrs = 0;
|
||
do {
|
||
var k = this.deterministicK(wif, hash, badrs);
|
||
var G = curve.getG();
|
||
var Q = G.multiply(k);
|
||
var r = Q.getX().toBigInteger().mod(n);
|
||
var s = k
|
||
.modInverse(n)
|
||
.multiply(e.add(priv.multiply(r)))
|
||
.mod(n);
|
||
badrs++;
|
||
} while (
|
||
r.compareTo(BigInteger.ZERO) <= 0 ||
|
||
s.compareTo(BigInteger.ZERO) <= 0
|
||
);
|
||
|
||
// Force lower s values per BIP62
|
||
var halfn = n.shiftRight(1);
|
||
if (s.compareTo(halfn) > 0) {
|
||
s = n.subtract(s);
|
||
}
|
||
|
||
var sig = serializeSig(r, s);
|
||
sig.push(parseInt(shType, 10));
|
||
|
||
return Crypto.util.bytesToHex(sig);
|
||
} else {
|
||
return false;
|
||
}
|
||
};
|
||
|
||
// https://tools.ietf.org/html/rfc6979#section-3.2
|
||
btrx.deterministicK = function (wif, hash, badrs) {
|
||
// if r or s were invalid when this function was used in signing,
|
||
// we do not want to actually compute r, s here for efficiency, so,
|
||
// we can increment badrs. explained at end of RFC 6979 section 3.2
|
||
|
||
// wif is b58check encoded wif privkey.
|
||
// hash is byte array of transaction digest.
|
||
// badrs is used only if the k resulted in bad r or s.
|
||
|
||
// some necessary things out of the way for clarity.
|
||
badrs = badrs || 0;
|
||
var key = bitjs.wif2privkey(wif);
|
||
var x = Crypto.util.hexToBytes(key["privkey"]);
|
||
var curve = EllipticCurve.getSECCurveByName("secp256k1");
|
||
var N = curve.getN();
|
||
|
||
// Step: a
|
||
// hash is a byteArray of the message digest. so h1 == hash in our case
|
||
|
||
// Step: b
|
||
var v = [
|
||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||
1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||
];
|
||
|
||
// Step: c
|
||
var k = [
|
||
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||
0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||
];
|
||
|
||
// Step: d
|
||
k = Crypto.HMAC(
|
||
Crypto.SHA256,
|
||
v.concat([0]).concat(x).concat(hash),
|
||
k,
|
||
{
|
||
asBytes: true,
|
||
}
|
||
);
|
||
|
||
// Step: e
|
||
v = Crypto.HMAC(Crypto.SHA256, v, k, {
|
||
asBytes: true,
|
||
});
|
||
|
||
// Step: f
|
||
k = Crypto.HMAC(
|
||
Crypto.SHA256,
|
||
v.concat([1]).concat(x).concat(hash),
|
||
k,
|
||
{
|
||
asBytes: true,
|
||
}
|
||
);
|
||
|
||
// Step: g
|
||
v = Crypto.HMAC(Crypto.SHA256, v, k, {
|
||
asBytes: true,
|
||
});
|
||
|
||
// Step: h1
|
||
var T = [];
|
||
|
||
// Step: h2 (since we know tlen = qlen, just copy v to T.)
|
||
v = Crypto.HMAC(Crypto.SHA256, v, k, {
|
||
asBytes: true,
|
||
});
|
||
T = v;
|
||
|
||
// Step: h3
|
||
var KBigInt = BigInteger.fromByteArrayUnsigned(T);
|
||
|
||
// loop if KBigInt is not in the range of [1, N-1] or if badrs needs incrementing.
|
||
var i = 0;
|
||
while (
|
||
KBigInt.compareTo(N) >= 0 ||
|
||
KBigInt.compareTo(BigInteger.ZERO) <= 0 ||
|
||
i < badrs
|
||
) {
|
||
k = Crypto.HMAC(Crypto.SHA256, v.concat([0]), k, {
|
||
asBytes: true,
|
||
});
|
||
v = Crypto.HMAC(Crypto.SHA256, v, k, {
|
||
asBytes: true,
|
||
});
|
||
v = Crypto.HMAC(Crypto.SHA256, v, k, {
|
||
asBytes: true,
|
||
});
|
||
T = v;
|
||
KBigInt = BigInteger.fromByteArrayUnsigned(T);
|
||
i++;
|
||
}
|
||
|
||
return KBigInt;
|
||
};
|
||
|
||
/* sign a "standard" input */
|
||
btrx.signinput = function (index, wif, sigHashType) {
|
||
var key = bitjs.wif2pubkey(wif);
|
||
var shType = sigHashType || 1;
|
||
var signature = this.transactionSig(index, wif, shType);
|
||
var buf = [];
|
||
var sigBytes = Crypto.util.hexToBytes(signature);
|
||
buf.push(sigBytes.length);
|
||
buf = buf.concat(sigBytes);
|
||
var pubKeyBytes = Crypto.util.hexToBytes(key["pubkey"]);
|
||
buf.push(pubKeyBytes.length);
|
||
buf = buf.concat(pubKeyBytes);
|
||
this.inputs[index].script = buf;
|
||
return true;
|
||
};
|
||
|
||
/* sign inputs */
|
||
btrx.sign = function (wif, sigHashType) {
|
||
var shType = sigHashType || 1;
|
||
for (var i = 0; i < this.inputs.length; i++) {
|
||
this.signinput(i, wif, shType);
|
||
}
|
||
return this.serialize();
|
||
};
|
||
|
||
/* serialize a transaction */
|
||
btrx.serialize = function () {
|
||
var buffer = [];
|
||
buffer = buffer.concat(bitjs.numToBytes(parseInt(this.version), 4));
|
||
|
||
buffer = buffer.concat(bitjs.numToVarInt(this.inputs.length));
|
||
for (var i = 0; i < this.inputs.length; i++) {
|
||
var txin = this.inputs[i];
|
||
buffer = buffer.concat(
|
||
Crypto.util.hexToBytes(txin.outpoint.hash).reverse()
|
||
);
|
||
buffer = buffer.concat(
|
||
bitjs.numToBytes(parseInt(txin.outpoint.index), 4)
|
||
);
|
||
var scriptBytes = txin.script;
|
||
buffer = buffer.concat(bitjs.numToVarInt(scriptBytes.length));
|
||
buffer = buffer.concat(scriptBytes);
|
||
buffer = buffer.concat(
|
||
bitjs.numToBytes(parseInt(txin.sequence), 4)
|
||
);
|
||
}
|
||
buffer = buffer.concat(bitjs.numToVarInt(this.outputs.length));
|
||
|
||
for (var i = 0; i < this.outputs.length; i++) {
|
||
var txout = this.outputs[i];
|
||
buffer = buffer.concat(bitjs.numToBytes(txout.value, 8));
|
||
var scriptBytes = txout.script;
|
||
buffer = buffer.concat(bitjs.numToVarInt(scriptBytes.length));
|
||
buffer = buffer.concat(scriptBytes);
|
||
}
|
||
|
||
buffer = buffer.concat(bitjs.numToBytes(parseInt(this.locktime), 4));
|
||
flohex = ascii_to_hexa(this.floData);
|
||
floDataCount = this.floData.length;
|
||
|
||
//flochange -- creating unique data character count logic for floData. This string is prefixed before actual floData string in Raw Transaction
|
||
if (floDataCount <= 16) {
|
||
floDataCountString = floDataCount.toString(16);
|
||
floDataCountString = "0" + floDataCountString;
|
||
} else if (floDataCount < 253) {
|
||
floDataCountString = floDataCount.toString(16);
|
||
} else if (floDataCount <= 1023) {
|
||
floDataCountAdjusted = floDataCount - 253 + parseInt("0xfd00fd");
|
||
floDataCountStringAdjusted = floDataCountAdjusted.toString(16);
|
||
floDataCountString =
|
||
floDataCountStringAdjusted.substr(0, 2) +
|
||
floDataCountStringAdjusted.substr(4, 2) +
|
||
floDataCountStringAdjusted.substr(2, 2);
|
||
} else {
|
||
floDataCountString = "Character Limit Exceeded";
|
||
}
|
||
|
||
return Crypto.util.bytesToHex(buffer) + floDataCountString + flohex; // flochange -- Addition of floDataCountString and floData in serialization
|
||
};
|
||
|
||
return btrx;
|
||
};
|
||
|
||
bitjs.numToBytes = function (num, bytes) {
|
||
if (typeof bytes === "undefined") bytes = 8;
|
||
if (bytes == 0) {
|
||
return [];
|
||
} else if (num == -1) {
|
||
return Crypto.util.hexToBytes("ffffffffffffffff");
|
||
} else {
|
||
return [num % 256].concat(
|
||
bitjs.numToBytes(Math.floor(num / 256), bytes - 1)
|
||
);
|
||
}
|
||
};
|
||
|
||
bitjs.numToByteArray = function (num) {
|
||
if (num <= 256) {
|
||
return [num];
|
||
} else {
|
||
return [num % 256].concat(
|
||
bitjs.numToByteArray(Math.floor(num / 256))
|
||
);
|
||
}
|
||
};
|
||
|
||
bitjs.numToVarInt = function (num) {
|
||
if (num < 253) {
|
||
return [num];
|
||
} else if (num < 65536) {
|
||
return [253].concat(bitjs.numToBytes(num, 2));
|
||
} else if (num < 4294967296) {
|
||
return [254].concat(bitjs.numToBytes(num, 4));
|
||
} else {
|
||
return [255].concat(bitjs.numToBytes(num, 8));
|
||
}
|
||
};
|
||
|
||
bitjs.bytesToNum = function (bytes) {
|
||
if (bytes.length == 0) return 0;
|
||
else return bytes[0] + 256 * bitjs.bytesToNum(bytes.slice(1));
|
||
};
|
||
|
||
/* clone an object */
|
||
bitjs.clone = function (obj) {
|
||
if (obj == null || typeof obj != "object") return obj;
|
||
var temp = new obj.constructor();
|
||
|
||
for (var key in obj) {
|
||
if (obj.hasOwnProperty(key)) {
|
||
temp[key] = bitjs.clone(obj[key]);
|
||
}
|
||
}
|
||
return temp;
|
||
};
|
||
|
||
var B58 = (bitjs.Base58 = {
|
||
alphabet: "123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz",
|
||
validRegex: /^[1-9A-HJ-NP-Za-km-z]+$/,
|
||
base: BigInteger.valueOf(58),
|
||
|
||
/**
|
||
* Convert a byte array to a base58-encoded string.
|
||
*
|
||
* Written by Mike Hearn for BitcoinJ.
|
||
* Copyright (c) 2011 Google Inc.
|
||
*
|
||
* Ported to JavaScript by Stefan Thomas.
|
||
*/
|
||
encode: function (input) {
|
||
var bi = BigInteger.fromByteArrayUnsigned(input);
|
||
var chars = [];
|
||
|
||
while (bi.compareTo(B58.base) >= 0) {
|
||
var mod = bi.mod(B58.base);
|
||
chars.unshift(B58.alphabet[mod.intValue()]);
|
||
bi = bi.subtract(mod).divide(B58.base);
|
||
}
|
||
chars.unshift(B58.alphabet[bi.intValue()]);
|
||
|
||
// Convert leading zeros too.
|
||
for (var i = 0; i < input.length; i++) {
|
||
if (input[i] == 0x00) {
|
||
chars.unshift(B58.alphabet[0]);
|
||
} else break;
|
||
}
|
||
|
||
return chars.join("");
|
||
},
|
||
|
||
/**
|
||
* Convert a base58-encoded string to a byte array.
|
||
*
|
||
* Written by Mike Hearn for BitcoinJ.
|
||
* Copyright (c) 2011 Google Inc.
|
||
*
|
||
* Ported to JavaScript by Stefan Thomas.
|
||
*/
|
||
decode: function (input) {
|
||
var bi = BigInteger.valueOf(0);
|
||
var leadingZerosNum = 0;
|
||
for (var i = input.length - 1; i >= 0; i--) {
|
||
var alphaIndex = B58.alphabet.indexOf(input[i]);
|
||
if (alphaIndex < 0) {
|
||
throw "Invalid character";
|
||
}
|
||
bi = bi.add(
|
||
BigInteger.valueOf(alphaIndex).multiply(
|
||
B58.base.pow(input.length - 1 - i)
|
||
)
|
||
);
|
||
|
||
// This counts leading zero bytes
|
||
if (input[i] == "1") leadingZerosNum++;
|
||
else leadingZerosNum = 0;
|
||
}
|
||
var bytes = bi.toByteArrayUnsigned();
|
||
|
||
// Add leading zeros
|
||
while (leadingZerosNum-- > 0) bytes.unshift(0);
|
||
|
||
return bytes;
|
||
},
|
||
});
|
||
return bitjs;
|
||
})();
|
||
|
||
/*
|
||
Copyright (c) 2011 Stefan Thomas
|
||
|
||
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
||
|
||
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
|
||
|
||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||
*/
|
||
|
||
//https://raw.github.com/bitcoinjs/bitcoinjs-lib/1a7fc9d063f864058809d06ef4542af40be3558f/src/bitcoin.js
|
||
(function (exports) {
|
||
var Bitcoin = exports;
|
||
})("object" === typeof module ? module.exports : (window.Bitcoin = {}));
|
||
|
||
//https://raw.github.com/bitcoinjs/bitcoinjs-lib/c952aaeb3ee472e3776655b8ea07299ebed702c7/src/base58.js
|
||
(function (Bitcoin) {
|
||
Bitcoin.Base58 = {
|
||
alphabet: "123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz",
|
||
validRegex: /^[1-9A-HJ-NP-Za-km-z]+$/,
|
||
base: BigInteger.valueOf(58),
|
||
|
||
/**
|
||
* Convert a byte array to a base58-encoded string.
|
||
*
|
||
* Written by Mike Hearn for BitcoinJ.
|
||
* Copyright (c) 2011 Google Inc.
|
||
*
|
||
* Ported to JavaScript by Stefan Thomas.
|
||
*/
|
||
encode: function (input) {
|
||
var bi = BigInteger.fromByteArrayUnsigned(input);
|
||
var chars = [];
|
||
|
||
while (bi.compareTo(B58.base) >= 0) {
|
||
var mod = bi.mod(B58.base);
|
||
chars.unshift(B58.alphabet[mod.intValue()]);
|
||
bi = bi.subtract(mod).divide(B58.base);
|
||
}
|
||
chars.unshift(B58.alphabet[bi.intValue()]);
|
||
|
||
// Convert leading zeros too.
|
||
for (var i = 0; i < input.length; i++) {
|
||
if (input[i] == 0x00) {
|
||
chars.unshift(B58.alphabet[0]);
|
||
} else break;
|
||
}
|
||
|
||
return chars.join("");
|
||
},
|
||
|
||
/**
|
||
* Convert a base58-encoded string to a byte array.
|
||
*
|
||
* Written by Mike Hearn for BitcoinJ.
|
||
* Copyright (c) 2011 Google Inc.
|
||
*
|
||
* Ported to JavaScript by Stefan Thomas.
|
||
*/
|
||
decode: function (input) {
|
||
var bi = BigInteger.valueOf(0);
|
||
var leadingZerosNum = 0;
|
||
for (var i = input.length - 1; i >= 0; i--) {
|
||
var alphaIndex = B58.alphabet.indexOf(input[i]);
|
||
if (alphaIndex < 0) {
|
||
throw "Invalid character";
|
||
}
|
||
bi = bi.add(
|
||
BigInteger.valueOf(alphaIndex).multiply(
|
||
B58.base.pow(input.length - 1 - i)
|
||
)
|
||
);
|
||
|
||
// This counts leading zero bytes
|
||
if (input[i] == "1") leadingZerosNum++;
|
||
else leadingZerosNum = 0;
|
||
}
|
||
var bytes = bi.toByteArrayUnsigned();
|
||
|
||
// Add leading zeros
|
||
while (leadingZerosNum-- > 0) bytes.unshift(0);
|
||
|
||
return bytes;
|
||
},
|
||
};
|
||
|
||
var B58 = Bitcoin.Base58;
|
||
})("undefined" != typeof Bitcoin ? Bitcoin : module.exports);
|
||
//https://raw.github.com/bitcoinjs/bitcoinjs-lib/09e8c6e184d6501a0c2c59d73ca64db5c0d3eb95/src/address.js
|
||
Bitcoin.Address = function (bytes) {
|
||
if (floGlobals.blockchain == "FLO") this.version = 0x23;
|
||
// FLO mainnet public address
|
||
else if (floGlobals.blockchain == "FLO_TEST") this.version = 0x73; // FLO testnet public address
|
||
if ("string" == typeof bytes) {
|
||
bytes = Bitcoin.Address.decodeString(bytes, this.version);
|
||
}
|
||
this.hash = bytes;
|
||
};
|
||
|
||
Bitcoin.Address.networkVersion = 0x23; // (FLO mainnet 0x23, 35D), (Bitcoin Mainnet, 0x00, 0D) // *this has no effect *
|
||
|
||
/**
|
||
* Serialize this object as a standard Bitcoin address.
|
||
*
|
||
* Returns the address as a base58-encoded string in the standardized format.
|
||
*/
|
||
Bitcoin.Address.prototype.toString = function () {
|
||
// Get a copy of the hash
|
||
var hash = this.hash.slice(0);
|
||
|
||
// Version
|
||
hash.unshift(this.version);
|
||
var checksum = Crypto.SHA256(
|
||
Crypto.SHA256(hash, {
|
||
asBytes: true,
|
||
}),
|
||
{
|
||
asBytes: true,
|
||
}
|
||
);
|
||
var bytes = hash.concat(checksum.slice(0, 4));
|
||
return Bitcoin.Base58.encode(bytes);
|
||
};
|
||
|
||
Bitcoin.Address.prototype.getHashBase64 = function () {
|
||
return Crypto.util.bytesToBase64(this.hash);
|
||
};
|
||
|
||
/**
|
||
* Parse a Bitcoin address contained in a string.
|
||
*/
|
||
Bitcoin.Address.decodeString = function (string, version) {
|
||
var bytes = Bitcoin.Base58.decode(string);
|
||
var hash = bytes.slice(0, 21);
|
||
var checksum = Crypto.SHA256(
|
||
Crypto.SHA256(hash, {
|
||
asBytes: true,
|
||
}),
|
||
{
|
||
asBytes: true,
|
||
}
|
||
);
|
||
|
||
if (
|
||
checksum[0] != bytes[21] ||
|
||
checksum[1] != bytes[22] ||
|
||
checksum[2] != bytes[23] ||
|
||
checksum[3] != bytes[24]
|
||
) {
|
||
throw "Checksum validation failed!";
|
||
}
|
||
|
||
if (version != hash.shift()) {
|
||
throw "Version " + hash.shift() + " not supported!";
|
||
}
|
||
|
||
return hash;
|
||
};
|
||
//https://raw.github.com/bitcoinjs/bitcoinjs-lib/e90780d3d3b8fc0d027d2bcb38b80479902f223e/src/ecdsa.js
|
||
Bitcoin.ECDSA = (function () {
|
||
var ecparams = EllipticCurve.getSECCurveByName("secp256k1");
|
||
var rng = new SecureRandom();
|
||
|
||
var P_OVER_FOUR = null;
|
||
|
||
function implShamirsTrick(P, k, Q, l) {
|
||
var m = Math.max(k.bitLength(), l.bitLength());
|
||
var Z = P.add2D(Q);
|
||
var R = P.curve.getInfinity();
|
||
|
||
for (var i = m - 1; i >= 0; --i) {
|
||
R = R.twice2D();
|
||
|
||
R.z = BigInteger.ONE;
|
||
|
||
if (k.testBit(i)) {
|
||
if (l.testBit(i)) {
|
||
R = R.add2D(Z);
|
||
} else {
|
||
R = R.add2D(P);
|
||
}
|
||
} else {
|
||
if (l.testBit(i)) {
|
||
R = R.add2D(Q);
|
||
}
|
||
}
|
||
}
|
||
|
||
return R;
|
||
}
|
||
|
||
var ECDSA = {
|
||
getBigRandom: function (limit) {
|
||
return new BigInteger(limit.bitLength(), rng)
|
||
.mod(limit.subtract(BigInteger.ONE))
|
||
.add(BigInteger.ONE);
|
||
},
|
||
sign: function (hash, priv) {
|
||
var d = priv;
|
||
var n = ecparams.getN();
|
||
var e = BigInteger.fromByteArrayUnsigned(hash);
|
||
|
||
do {
|
||
var k = ECDSA.getBigRandom(n);
|
||
var G = ecparams.getG();
|
||
var Q = G.multiply(k);
|
||
var r = Q.getX().toBigInteger().mod(n);
|
||
} while (r.compareTo(BigInteger.ZERO) <= 0);
|
||
|
||
var s = k
|
||
.modInverse(n)
|
||
.multiply(e.add(d.multiply(r)))
|
||
.mod(n);
|
||
|
||
return ECDSA.serializeSig(r, s);
|
||
},
|
||
|
||
verify: function (hash, sig, pubkey) {
|
||
var r, s;
|
||
if (Bitcoin.Util.isArray(sig)) {
|
||
var obj = ECDSA.parseSig(sig);
|
||
r = obj.r;
|
||
s = obj.s;
|
||
} else if ("object" === typeof sig && sig.r && sig.s) {
|
||
r = sig.r;
|
||
s = sig.s;
|
||
} else {
|
||
throw "Invalid value for signature";
|
||
}
|
||
|
||
var Q;
|
||
if (pubkey instanceof ec.PointFp) {
|
||
Q = pubkey;
|
||
} else if (Bitcoin.Util.isArray(pubkey)) {
|
||
Q = EllipticCurve.PointFp.decodeFrom(ecparams.getCurve(), pubkey);
|
||
} else {
|
||
throw "Invalid format for pubkey value, must be byte array or ec.PointFp";
|
||
}
|
||
var e = BigInteger.fromByteArrayUnsigned(hash);
|
||
|
||
return ECDSA.verifyRaw(e, r, s, Q);
|
||
},
|
||
|
||
verifyRaw: function (e, r, s, Q) {
|
||
var n = ecparams.getN();
|
||
var G = ecparams.getG();
|
||
|
||
if (r.compareTo(BigInteger.ONE) < 0 || r.compareTo(n) >= 0)
|
||
return false;
|
||
|
||
if (s.compareTo(BigInteger.ONE) < 0 || s.compareTo(n) >= 0)
|
||
return false;
|
||
|
||
var c = s.modInverse(n);
|
||
|
||
var u1 = e.multiply(c).mod(n);
|
||
var u2 = r.multiply(c).mod(n);
|
||
|
||
// TODO(!!!): For some reason Shamir's trick isn't working with
|
||
// signed message verification!? Probably an implementation
|
||
// error!
|
||
//var point = implShamirsTrick(G, u1, Q, u2);
|
||
var point = G.multiply(u1).add(Q.multiply(u2));
|
||
|
||
var v = point.getX().toBigInteger().mod(n);
|
||
|
||
return v.equals(r);
|
||
},
|
||
|
||
/**
|
||
* Serialize a signature into DER format.
|
||
*
|
||
* Takes two BigIntegers representing r and s and returns a byte array.
|
||
*/
|
||
serializeSig: function (r, s) {
|
||
var rBa = r.toByteArraySigned();
|
||
var sBa = s.toByteArraySigned();
|
||
|
||
var sequence = [];
|
||
sequence.push(0x02); // INTEGER
|
||
sequence.push(rBa.length);
|
||
sequence = sequence.concat(rBa);
|
||
|
||
sequence.push(0x02); // INTEGER
|
||
sequence.push(sBa.length);
|
||
sequence = sequence.concat(sBa);
|
||
|
||
sequence.unshift(sequence.length);
|
||
sequence.unshift(0x30); // SEQUENCE
|
||
|
||
return sequence;
|
||
},
|
||
|
||
/**
|
||
* Parses a byte array containing a DER-encoded signature.
|
||
*
|
||
* This function will return an object of the form:
|
||
*
|
||
* {
|
||
* r: BigInteger,
|
||
* s: BigInteger
|
||
* }
|
||
*/
|
||
parseSig: function (sig) {
|
||
var cursor;
|
||
if (sig[0] != 0x30)
|
||
throw new Error("Signature not a valid DERSequence");
|
||
|
||
cursor = 2;
|
||
if (sig[cursor] != 0x02)
|
||
throw new Error("First element in signature must be a DERInteger");
|
||
var rBa = sig.slice(cursor + 2, cursor + 2 + sig[cursor + 1]);
|
||
|
||
cursor += 2 + sig[cursor + 1];
|
||
if (sig[cursor] != 0x02)
|
||
throw new Error("Second element in signature must be a DERInteger");
|
||
var sBa = sig.slice(cursor + 2, cursor + 2 + sig[cursor + 1]);
|
||
|
||
cursor += 2 + sig[cursor + 1];
|
||
|
||
//if (cursor != sig.length)
|
||
// throw new Error("Extra bytes in signature");
|
||
|
||
var r = BigInteger.fromByteArrayUnsigned(rBa);
|
||
var s = BigInteger.fromByteArrayUnsigned(sBa);
|
||
|
||
return {
|
||
r: r,
|
||
s: s,
|
||
};
|
||
},
|
||
|
||
parseSigCompact: function (sig) {
|
||
if (sig.length !== 65) {
|
||
throw "Signature has the wrong length";
|
||
}
|
||
|
||
// Signature is prefixed with a type byte storing three bits of
|
||
// information.
|
||
var i = sig[0] - 27;
|
||
if (i < 0 || i > 7) {
|
||
throw "Invalid signature type";
|
||
}
|
||
|
||
var n = ecparams.getN();
|
||
var r = BigInteger.fromByteArrayUnsigned(sig.slice(1, 33)).mod(n);
|
||
var s = BigInteger.fromByteArrayUnsigned(sig.slice(33, 65)).mod(n);
|
||
|
||
return {
|
||
r: r,
|
||
s: s,
|
||
i: i,
|
||
};
|
||
},
|
||
|
||
/**
|
||
* Recover a public key from a signature.
|
||
*
|
||
* See SEC 1: Elliptic Curve Cryptography, section 4.1.6, "Public
|
||
* Key Recovery Operation".
|
||
*
|
||
* http://www.secg.org/download/aid-780/sec1-v2.pdf
|
||
*/
|
||
recoverPubKey: function (r, s, hash, i) {
|
||
// The recovery parameter i has two bits.
|
||
i = i & 3;
|
||
|
||
// The less significant bit specifies whether the y coordinate
|
||
// of the compressed point is even or not.
|
||
var isYEven = i & 1;
|
||
|
||
// The more significant bit specifies whether we should use the
|
||
// first or second candidate key.
|
||
var isSecondKey = i >> 1;
|
||
|
||
var n = ecparams.getN();
|
||
var G = ecparams.getG();
|
||
var curve = ecparams.getCurve();
|
||
var p = curve.getQ();
|
||
var a = curve.getA().toBigInteger();
|
||
var b = curve.getB().toBigInteger();
|
||
|
||
// We precalculate (p + 1) / 4 where p is if the field order
|
||
if (!P_OVER_FOUR) {
|
||
P_OVER_FOUR = p.add(BigInteger.ONE).divide(BigInteger.valueOf(4));
|
||
}
|
||
|
||
// 1.1 Compute x
|
||
var x = isSecondKey ? r.add(n) : r;
|
||
|
||
// 1.3 Convert x to point
|
||
var alpha = x
|
||
.multiply(x)
|
||
.multiply(x)
|
||
.add(a.multiply(x))
|
||
.add(b)
|
||
.mod(p);
|
||
var beta = alpha.modPow(P_OVER_FOUR, p);
|
||
|
||
var xorOdd = beta.isEven() ? i % 2 : (i + 1) % 2;
|
||
// If beta is even, but y isn't or vice versa, then convert it,
|
||
// otherwise we're done and y == beta.
|
||
var y = (beta.isEven() ? !isYEven : isYEven)
|
||
? beta
|
||
: p.subtract(beta);
|
||
|
||
// 1.4 Check that nR is at infinity
|
||
var R = new EllipticCurve.PointFp(
|
||
curve,
|
||
curve.fromBigInteger(x),
|
||
curve.fromBigInteger(y)
|
||
);
|
||
R.validate();
|
||
|
||
// 1.5 Compute e from M
|
||
var e = BigInteger.fromByteArrayUnsigned(hash);
|
||
var eNeg = BigInteger.ZERO.subtract(e).mod(n);
|
||
|
||
// 1.6 Compute Q = r^-1 (sR - eG)
|
||
var rInv = r.modInverse(n);
|
||
var Q = implShamirsTrick(R, s, G, eNeg).multiply(rInv);
|
||
|
||
Q.validate();
|
||
if (!ECDSA.verifyRaw(e, r, s, Q)) {
|
||
throw "Pubkey recovery unsuccessful";
|
||
}
|
||
|
||
var pubKey = new Bitcoin.ECKey();
|
||
pubKey.pub = Q;
|
||
return pubKey;
|
||
},
|
||
|
||
/**
|
||
* Calculate pubkey extraction parameter.
|
||
*
|
||
* When extracting a pubkey from a signature, we have to
|
||
* distinguish four different cases. Rather than putting this
|
||
* burden on the verifier, Bitcoin includes a 2-bit value with the
|
||
* signature.
|
||
*
|
||
* This function simply tries all four cases and returns the value
|
||
* that resulted in a successful pubkey recovery.
|
||
*/
|
||
calcPubkeyRecoveryParam: function (address, r, s, hash) {
|
||
for (var i = 0; i < 4; i++) {
|
||
try {
|
||
var pubkey = Bitcoin.ECDSA.recoverPubKey(r, s, hash, i);
|
||
if (pubkey.getBitcoinAddress().toString() == address) {
|
||
return i;
|
||
}
|
||
} catch (e) {}
|
||
}
|
||
throw "Unable to find valid recovery factor";
|
||
},
|
||
};
|
||
|
||
return ECDSA;
|
||
})();
|
||
Bitcoin.KeyPool = (function () {
|
||
var KeyPool = function () {
|
||
this.keyArray = [];
|
||
|
||
this.push = function (item) {
|
||
if (item == null || item.priv == null) return;
|
||
var doAdd = true;
|
||
// prevent duplicates from being added to the array
|
||
for (var index in this.keyArray) {
|
||
var currentItem = this.keyArray[index];
|
||
if (
|
||
currentItem != null &&
|
||
currentItem.priv != null &&
|
||
item.getBitcoinAddress() == currentItem.getBitcoinAddress()
|
||
) {
|
||
doAdd = false;
|
||
break;
|
||
}
|
||
}
|
||
if (doAdd) this.keyArray.push(item);
|
||
};
|
||
|
||
this.reset = function () {
|
||
this.keyArray = [];
|
||
};
|
||
|
||
this.getArray = function () {
|
||
// copy array
|
||
return this.keyArray.slice(0);
|
||
};
|
||
|
||
this.setArray = function (ka) {
|
||
this.keyArray = ka;
|
||
};
|
||
|
||
this.length = function () {
|
||
return this.keyArray.length;
|
||
};
|
||
|
||
this.toString = function () {
|
||
var keyPoolString = "# = " + this.length() + "\n";
|
||
var pool = this.getArray();
|
||
for (var index in pool) {
|
||
var item = pool[index];
|
||
if (
|
||
Bitcoin.Util.hasMethods(item, "getBitcoinAddress", "toString")
|
||
) {
|
||
if (item != null) {
|
||
keyPoolString +=
|
||
'"' +
|
||
item.getBitcoinAddress() +
|
||
'"' +
|
||
', "' +
|
||
item.toString("wif") +
|
||
'"\n';
|
||
}
|
||
}
|
||
}
|
||
|
||
return keyPoolString;
|
||
};
|
||
|
||
return this;
|
||
};
|
||
|
||
return new KeyPool();
|
||
})();
|
||
|
||
Bitcoin.Bip38Key = (function () {
|
||
var Bip38 = function (address, encryptedKey) {
|
||
this.address = address;
|
||
this.priv = encryptedKey;
|
||
};
|
||
|
||
Bip38.prototype.getBitcoinAddress = function () {
|
||
return this.address;
|
||
};
|
||
|
||
Bip38.prototype.toString = function () {
|
||
return this.priv;
|
||
};
|
||
|
||
return Bip38;
|
||
})();
|
||
|
||
//https://raw.github.com/pointbiz/bitcoinjs-lib/9b2f94a028a7bc9bed94e0722563e9ff1d8e8db8/src/eckey.js
|
||
Bitcoin.ECKey = (function () {
|
||
var ECDSA = Bitcoin.ECDSA;
|
||
var KeyPool = Bitcoin.KeyPool;
|
||
var ecparams = EllipticCurve.getSECCurveByName("secp256k1");
|
||
|
||
var ECKey = function (input) {
|
||
if (!input) {
|
||
// Generate new key
|
||
var n = ecparams.getN();
|
||
this.priv = ECDSA.getBigRandom(n);
|
||
} else if (input instanceof BigInteger) {
|
||
// Input is a private key value
|
||
this.priv = input;
|
||
} else if (Bitcoin.Util.isArray(input)) {
|
||
// Prepend zero byte to prevent interpretation as negative integer
|
||
this.priv = BigInteger.fromByteArrayUnsigned(input);
|
||
} else if ("string" == typeof input) {
|
||
var bytes = null;
|
||
try {
|
||
// This part is edited for FLO. FLO WIF are always compressed WIF. FLO WIF (private key) starts with R for mainnet and c for testnet.
|
||
if (
|
||
(floGlobals.blockchain == "FLO" &&
|
||
/^R[123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{51}$/.test(
|
||
input
|
||
)) ||
|
||
(floGlobals.blockchain == "FLO_TEST" &&
|
||
/^c[123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{51}$/.test(
|
||
input
|
||
))
|
||
) {
|
||
bytes = ECKey.decodeCompressedWalletImportFormat(input);
|
||
this.compressed = true;
|
||
} else if (ECKey.isHexFormat(input)) {
|
||
bytes = Crypto.util.hexToBytes(input);
|
||
}
|
||
|
||
/*
|
||
if (ECKey.isWalletImportFormat(input)) {
|
||
bytes = ECKey.decodeWalletImportFormat(input);
|
||
} else if (ECKey.isCompressedWalletImportFormat(input)) {
|
||
bytes = ECKey.decodeCompressedWalletImportFormat(input);
|
||
this.compressed = true;
|
||
} else if (ECKey.isMiniFormat(input)) {
|
||
bytes = Crypto.SHA256(input, { asBytes: true });
|
||
} else if (ECKey.isHexFormat(input)) {
|
||
bytes = Crypto.util.hexToBytes(input);
|
||
} else if (ECKey.isBase64Format(input)) {
|
||
bytes = Crypto.util.base64ToBytes(input);
|
||
}
|
||
*/
|
||
} catch (exc1) {
|
||
this.setError(exc1);
|
||
}
|
||
|
||
if (ECKey.isBase6Format(input)) {
|
||
this.priv = new BigInteger(input, 6);
|
||
} else if (bytes == null || bytes.length != 32) {
|
||
this.priv = null;
|
||
} else {
|
||
// Prepend zero byte to prevent interpretation as negative integer
|
||
this.priv = BigInteger.fromByteArrayUnsigned(bytes);
|
||
}
|
||
}
|
||
|
||
this.compressed =
|
||
this.compressed == undefined
|
||
? !!ECKey.compressByDefault
|
||
: this.compressed;
|
||
try {
|
||
// check not zero
|
||
if (this.priv != null && BigInteger.ZERO.compareTo(this.priv) == 0)
|
||
this.setError("Error: BigInteger equal to zero.");
|
||
// valid range [0x1, 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364140])
|
||
var hexKeyRangeLimit =
|
||
"FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364140";
|
||
var rangeLimitBytes = Crypto.util.hexToBytes(hexKeyRangeLimit);
|
||
var limitBigInt = BigInteger.fromByteArrayUnsigned(rangeLimitBytes);
|
||
if (this.priv != null && limitBigInt.compareTo(this.priv) < 0)
|
||
this.setError("Error: BigInteger outside of curve range.");
|
||
|
||
if (this.priv != null) {
|
||
KeyPool.push(this);
|
||
}
|
||
} catch (exc2) {
|
||
this.setError(exc2);
|
||
}
|
||
};
|
||
|
||
if (floGlobals.blockchain == "FLO") ECKey.privateKeyPrefix = 0xa3;
|
||
//(Bitcoin mainnet 0x80 testnet 0xEF) (FLO mainnet 0xA3 163 D)
|
||
else if (floGlobals.blockchain == "FLO_TEST")
|
||
ECKey.privateKeyPrefix = 0xef; //FLO testnet
|
||
|
||
/**
|
||
* Whether public keys should be returned compressed by default.
|
||
*/
|
||
ECKey.compressByDefault = false;
|
||
|
||
/**
|
||
* Set whether the public key should be returned compressed or not.
|
||
*/
|
||
ECKey.prototype.setError = function (err) {
|
||
this.error = err;
|
||
this.priv = null;
|
||
return this;
|
||
};
|
||
|
||
/**
|
||
* Set whether the public key should be returned compressed or not.
|
||
*/
|
||
ECKey.prototype.setCompressed = function (v) {
|
||
this.compressed = !!v;
|
||
if (this.pubPoint) this.pubPoint.compressed = this.compressed;
|
||
return this;
|
||
};
|
||
|
||
/*
|
||
* Return public key as a byte array in DER encoding
|
||
*/
|
||
ECKey.prototype.getPub = function () {
|
||
if (this.compressed) {
|
||
if (this.pubComp) return this.pubComp;
|
||
return (this.pubComp = this.getPubPoint().getEncoded(1));
|
||
} else {
|
||
if (this.pubUncomp) return this.pubUncomp;
|
||
return (this.pubUncomp = this.getPubPoint().getEncoded(0));
|
||
}
|
||
};
|
||
|
||
/**
|
||
* Return public point as ECPoint object.
|
||
*/
|
||
ECKey.prototype.getPubPoint = function () {
|
||
if (!this.pubPoint) {
|
||
this.pubPoint = ecparams.getG().multiply(this.priv);
|
||
this.pubPoint.compressed = this.compressed;
|
||
}
|
||
return this.pubPoint;
|
||
};
|
||
|
||
ECKey.prototype.getPubKeyHex = function () {
|
||
if (this.compressed) {
|
||
if (this.pubKeyHexComp) return this.pubKeyHexComp;
|
||
return (this.pubKeyHexComp = Crypto.util
|
||
.bytesToHex(this.getPub())
|
||
.toString()
|
||
.toUpperCase());
|
||
} else {
|
||
if (this.pubKeyHexUncomp) return this.pubKeyHexUncomp;
|
||
return (this.pubKeyHexUncomp = Crypto.util
|
||
.bytesToHex(this.getPub())
|
||
.toString()
|
||
.toUpperCase());
|
||
}
|
||
};
|
||
|
||
/**
|
||
* Get the pubKeyHash for this key.
|
||
*
|
||
* This is calculated as RIPE160(SHA256([encoded pubkey])) and returned as
|
||
* a byte array.
|
||
*/
|
||
ECKey.prototype.getPubKeyHash = function () {
|
||
if (this.compressed) {
|
||
if (this.pubKeyHashComp) return this.pubKeyHashComp;
|
||
return (this.pubKeyHashComp = Bitcoin.Util.sha256ripe160(
|
||
this.getPub()
|
||
));
|
||
} else {
|
||
if (this.pubKeyHashUncomp) return this.pubKeyHashUncomp;
|
||
return (this.pubKeyHashUncomp = Bitcoin.Util.sha256ripe160(
|
||
this.getPub()
|
||
));
|
||
}
|
||
};
|
||
|
||
ECKey.prototype.getBitcoinAddress = function () {
|
||
var hash = this.getPubKeyHash();
|
||
var addr = new Bitcoin.Address(hash);
|
||
return addr.toString();
|
||
};
|
||
|
||
/*
|
||
* Takes a public point as a hex string or byte array
|
||
*/
|
||
ECKey.prototype.setPub = function (pub) {
|
||
// byte array
|
||
if (Bitcoin.Util.isArray(pub)) {
|
||
pub = Crypto.util.bytesToHex(pub).toString().toUpperCase();
|
||
}
|
||
var ecPoint = ecparams.getCurve().decodePointHex(pub);
|
||
this.setCompressed(ecPoint.compressed);
|
||
this.pubPoint = ecPoint;
|
||
return this;
|
||
};
|
||
|
||
// Sipa Private Key Wallet Import Format
|
||
ECKey.prototype.getBitcoinWalletImportFormat = function () {
|
||
var bytes = this.getBitcoinPrivateKeyByteArray();
|
||
if (bytes == null) return "";
|
||
bytes.unshift(ECKey.privateKeyPrefix); // prepend 0x80 byte
|
||
if (this.compressed) bytes.push(0x01); // append 0x01 byte for compressed format
|
||
var checksum = Crypto.SHA256(
|
||
Crypto.SHA256(bytes, {
|
||
asBytes: true,
|
||
}),
|
||
{
|
||
asBytes: true,
|
||
}
|
||
);
|
||
bytes = bytes.concat(checksum.slice(0, 4));
|
||
var privWif = Bitcoin.Base58.encode(bytes);
|
||
return privWif;
|
||
};
|
||
|
||
// Private Key Hex Format
|
||
ECKey.prototype.getBitcoinHexFormat = function () {
|
||
return Crypto.util
|
||
.bytesToHex(this.getBitcoinPrivateKeyByteArray())
|
||
.toString()
|
||
.toUpperCase();
|
||
};
|
||
|
||
// Private Key Base64 Format
|
||
ECKey.prototype.getBitcoinBase64Format = function () {
|
||
return Crypto.util.bytesToBase64(this.getBitcoinPrivateKeyByteArray());
|
||
};
|
||
|
||
ECKey.prototype.getBitcoinPrivateKeyByteArray = function () {
|
||
if (this.priv == null) return null;
|
||
// Get a copy of private key as a byte array
|
||
var bytes = this.priv.toByteArrayUnsigned();
|
||
// zero pad if private key is less than 32 bytes
|
||
while (bytes.length < 32) bytes.unshift(0x00);
|
||
return bytes;
|
||
};
|
||
|
||
ECKey.prototype.toString = function (format) {
|
||
format = format || "";
|
||
if (
|
||
format.toString().toLowerCase() == "base64" ||
|
||
format.toString().toLowerCase() == "b64"
|
||
) {
|
||
return this.getBitcoinBase64Format();
|
||
}
|
||
// Wallet Import Format
|
||
else if (format.toString().toLowerCase() == "wif") {
|
||
return this.getBitcoinWalletImportFormat();
|
||
} else {
|
||
return this.getBitcoinHexFormat();
|
||
}
|
||
};
|
||
|
||
ECKey.prototype.sign = function (hash) {
|
||
return ECDSA.sign(hash, this.priv);
|
||
};
|
||
|
||
ECKey.prototype.verify = function (hash, sig) {
|
||
return ECDSA.verify(hash, sig, this.getPub());
|
||
};
|
||
|
||
/**
|
||
* Parse a wallet import format private key contained in a string.
|
||
*/
|
||
ECKey.decodeWalletImportFormat = function (privStr) {
|
||
var bytes = Bitcoin.Base58.decode(privStr);
|
||
var hash = bytes.slice(0, 33);
|
||
var checksum = Crypto.SHA256(
|
||
Crypto.SHA256(hash, {
|
||
asBytes: true,
|
||
}),
|
||
{
|
||
asBytes: true,
|
||
}
|
||
);
|
||
if (
|
||
checksum[0] != bytes[33] ||
|
||
checksum[1] != bytes[34] ||
|
||
checksum[2] != bytes[35] ||
|
||
checksum[3] != bytes[36]
|
||
) {
|
||
throw "Checksum validation failed!";
|
||
}
|
||
var version = hash.shift();
|
||
if (version != ECKey.privateKeyPrefix) {
|
||
throw "Version " + version + " not supported!";
|
||
}
|
||
return hash;
|
||
};
|
||
|
||
/**
|
||
* Parse a compressed wallet import format private key contained in a string.
|
||
*/
|
||
ECKey.decodeCompressedWalletImportFormat = function (privStr) {
|
||
var bytes = Bitcoin.Base58.decode(privStr);
|
||
var hash = bytes.slice(0, 34);
|
||
var checksum = Crypto.SHA256(
|
||
Crypto.SHA256(hash, {
|
||
asBytes: true,
|
||
}),
|
||
{
|
||
asBytes: true,
|
||
}
|
||
);
|
||
if (
|
||
checksum[0] != bytes[34] ||
|
||
checksum[1] != bytes[35] ||
|
||
checksum[2] != bytes[36] ||
|
||
checksum[3] != bytes[37]
|
||
) {
|
||
throw "Checksum validation failed!";
|
||
}
|
||
var version = hash.shift();
|
||
if (version != ECKey.privateKeyPrefix) {
|
||
throw "Version " + version + " not supported!";
|
||
}
|
||
hash.pop();
|
||
return hash;
|
||
};
|
||
|
||
// 64 characters [0-9A-F]
|
||
ECKey.isHexFormat = function (key) {
|
||
key = key.toString();
|
||
return /^[A-Fa-f0-9]{64}$/.test(key);
|
||
};
|
||
|
||
// 51 characters base58, always starts with a '5'
|
||
ECKey.isWalletImportFormat = function (key) {
|
||
key = key.toString();
|
||
return ECKey.privateKeyPrefix == 0x80
|
||
? /^5[123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{50}$/.test(
|
||
key
|
||
)
|
||
: /^R[123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{50}$/.test(
|
||
key
|
||
);
|
||
};
|
||
|
||
// 52 characters base58
|
||
ECKey.isCompressedWalletImportFormat = function (key) {
|
||
key = key.toString();
|
||
return ECKey.privateKeyPrefix == 0x80
|
||
? /^[LK][123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{51}$/.test(
|
||
key
|
||
)
|
||
: /^R[123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{51}$/.test(
|
||
key
|
||
);
|
||
};
|
||
|
||
// 44 characters
|
||
ECKey.isBase64Format = function (key) {
|
||
key = key.toString();
|
||
return /^[ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789=+\/]{44}$/.test(
|
||
key
|
||
);
|
||
};
|
||
|
||
// 99 characters, 1=1, if using dice convert 6 to 0
|
||
ECKey.isBase6Format = function (key) {
|
||
key = key.toString();
|
||
return /^[012345]{99}$/.test(key);
|
||
};
|
||
|
||
// 22, 26 or 30 characters, always starts with an 'S'
|
||
ECKey.isMiniFormat = function (key) {
|
||
key = key.toString();
|
||
var validChars22 =
|
||
/^S[123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{21}$/.test(
|
||
key
|
||
);
|
||
var validChars26 =
|
||
/^S[123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{25}$/.test(
|
||
key
|
||
);
|
||
var validChars30 =
|
||
/^S[123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{29}$/.test(
|
||
key
|
||
);
|
||
var testBytes = Crypto.SHA256(key + "?", {
|
||
asBytes: true,
|
||
});
|
||
|
||
return (
|
||
(testBytes[0] === 0x00 || testBytes[0] === 0x01) &&
|
||
(validChars22 || validChars26 || validChars30)
|
||
);
|
||
};
|
||
|
||
return ECKey;
|
||
})();
|
||
//https://raw.github.com/bitcoinjs/bitcoinjs-lib/09e8c6e184d6501a0c2c59d73ca64db5c0d3eb95/src/util.js
|
||
// Bitcoin utility functions
|
||
Bitcoin.Util = {
|
||
/**
|
||
* Cross-browser compatibility version of Array.isArray.
|
||
*/
|
||
isArray:
|
||
Array.isArray ||
|
||
function (o) {
|
||
return Object.prototype.toString.call(o) === "[object Array]";
|
||
},
|
||
/**
|
||
* Create an array of a certain length filled with a specific value.
|
||
*/
|
||
makeFilledArray: function (len, val) {
|
||
var array = [];
|
||
var i = 0;
|
||
while (i < len) {
|
||
array[i++] = val;
|
||
}
|
||
return array;
|
||
},
|
||
/**
|
||
* Turn an integer into a "var_int".
|
||
*
|
||
* "var_int" is a variable length integer used by Bitcoin's binary format.
|
||
*
|
||
* Returns a byte array.
|
||
*/
|
||
numToVarInt: function (i) {
|
||
if (i < 0xfd) {
|
||
// unsigned char
|
||
return [i];
|
||
} else if (i <= 1 << 16) {
|
||
// unsigned short (LE)
|
||
return [0xfd, i >>> 8, i & 255];
|
||
} else if (i <= 1 << 32) {
|
||
// unsigned int (LE)
|
||
return [0xfe].concat(Crypto.util.wordsToBytes([i]));
|
||
} else {
|
||
// unsigned long long (LE)
|
||
return [0xff].concat(Crypto.util.wordsToBytes([i >>> 32, i]));
|
||
}
|
||
},
|
||
/**
|
||
* Parse a Bitcoin value byte array, returning a BigInteger.
|
||
*/
|
||
valueToBigInt: function (valueBuffer) {
|
||
if (valueBuffer instanceof BigInteger) return valueBuffer;
|
||
|
||
// Prepend zero byte to prevent interpretation as negative integer
|
||
return BigInteger.fromByteArrayUnsigned(valueBuffer);
|
||
},
|
||
/**
|
||
* Format a Bitcoin value as a string.
|
||
*
|
||
* Takes a BigInteger or byte-array and returns that amount of Bitcoins in a
|
||
* nice standard formatting.
|
||
*
|
||
* Examples:
|
||
* 12.3555
|
||
* 0.1234
|
||
* 900.99998888
|
||
* 34.00
|
||
*/
|
||
formatValue: function (valueBuffer) {
|
||
var value = this.valueToBigInt(valueBuffer).toString();
|
||
var integerPart =
|
||
value.length > 8 ? value.substr(0, value.length - 8) : "0";
|
||
var decimalPart =
|
||
value.length > 8 ? value.substr(value.length - 8) : value;
|
||
while (decimalPart.length < 8) decimalPart = "0" + decimalPart;
|
||
decimalPart = decimalPart.replace(/0*$/, "");
|
||
while (decimalPart.length < 2) decimalPart += "0";
|
||
return integerPart + "." + decimalPart;
|
||
},
|
||
/**
|
||
* Parse a floating point string as a Bitcoin value.
|
||
*
|
||
* Keep in mind that parsing user input is messy. You should always display
|
||
* the parsed value back to the user to make sure we understood his input
|
||
* correctly.
|
||
*/
|
||
parseValue: function (valueString) {
|
||
// TODO: Detect other number formats (e.g. comma as decimal separator)
|
||
var valueComp = valueString.split(".");
|
||
var integralPart = valueComp[0];
|
||
var fractionalPart = valueComp[1] || "0";
|
||
while (fractionalPart.length < 8) fractionalPart += "0";
|
||
fractionalPart = fractionalPart.replace(/^0+/g, "");
|
||
var value = BigInteger.valueOf(parseInt(integralPart));
|
||
value = value.multiply(BigInteger.valueOf(100000000));
|
||
value = value.add(BigInteger.valueOf(parseInt(fractionalPart)));
|
||
return value;
|
||
},
|
||
/**
|
||
* Calculate RIPEMD160(SHA256(data)).
|
||
*
|
||
* Takes an arbitrary byte array as inputs and returns the hash as a byte
|
||
* array.
|
||
*/
|
||
sha256ripe160: function (data) {
|
||
return ripemd160(
|
||
Crypto.SHA256(data, {
|
||
asBytes: true,
|
||
}),
|
||
{
|
||
asBytes: true,
|
||
}
|
||
);
|
||
},
|
||
// double sha256
|
||
dsha256: function (data) {
|
||
return Crypto.SHA256(
|
||
Crypto.SHA256(data, {
|
||
asBytes: true,
|
||
}),
|
||
{
|
||
asBytes: true,
|
||
}
|
||
);
|
||
},
|
||
// duck typing method
|
||
hasMethods: function (obj /*, method list as strings */) {
|
||
var i = 1,
|
||
methodName;
|
||
while ((methodName = arguments[i++])) {
|
||
if (typeof obj[methodName] != "function") {
|
||
return false;
|
||
}
|
||
}
|
||
return true;
|
||
},
|
||
};
|
||
|
||
(function (ellipticCurveType) {
|
||
//Defining Elliptic Encryption Object
|
||
var ellipticEncryption = (window.ellipticCurveEncryption =
|
||
function () {});
|
||
|
||
ellipticEncryption.rng = new SecureRandom();
|
||
|
||
ellipticEncryption.getCurveParameters = function (curveName) {
|
||
//Default is secp256k1
|
||
curveName = typeof curveName !== "undefined" ? curveName : "secp256k1";
|
||
|
||
var c = EllipticCurve.getSECCurveByName(curveName);
|
||
var curveDetails = {
|
||
Q: "",
|
||
A: "",
|
||
B: "",
|
||
GX: "",
|
||
GY: "",
|
||
N: "",
|
||
};
|
||
|
||
curveDetails.Q = c.getCurve().getQ().toString();
|
||
curveDetails.A = c.getCurve().getA().toBigInteger().toString();
|
||
curveDetails.B = c.getCurve().getB().toBigInteger().toString();
|
||
curveDetails.GX = c.getG().getX().toBigInteger().toString();
|
||
curveDetails.GY = c.getG().getY().toBigInteger().toString();
|
||
curveDetails.N = c.getN().toString();
|
||
|
||
return curveDetails;
|
||
};
|
||
|
||
ellipticEncryption.selectedCurve =
|
||
ellipticEncryption.getCurveParameters(ellipticCurveType);
|
||
|
||
ellipticEncryption.get_curve = function () {
|
||
return new EllipticCurve.CurveFp(
|
||
new BigInteger(this.selectedCurve.Q),
|
||
new BigInteger(this.selectedCurve.A),
|
||
new BigInteger(this.selectedCurve.B)
|
||
);
|
||
};
|
||
|
||
ellipticEncryption.get_G = function (curve) {
|
||
return new EllipticCurve.PointFp(
|
||
curve,
|
||
curve.fromBigInteger(new BigInteger(this.selectedCurve.GX)),
|
||
curve.fromBigInteger(new BigInteger(this.selectedCurve.GY))
|
||
);
|
||
};
|
||
|
||
ellipticEncryption.pick_rand = function () {
|
||
var n = new BigInteger(this.selectedCurve.N);
|
||
var n1 = n.subtract(BigInteger.ONE);
|
||
var r = new BigInteger(n.bitLength(), this.rng);
|
||
return r.mod(n1).add(BigInteger.ONE);
|
||
};
|
||
|
||
ellipticEncryption.senderRandom = function () {
|
||
var r = this.pick_rand();
|
||
return r.toString();
|
||
};
|
||
|
||
ellipticEncryption.receiverRandom = function () {
|
||
//This is receivers private key. For now we will use random. CHANGE IT LATER
|
||
var r = this.pick_rand();
|
||
return r.toString();
|
||
};
|
||
|
||
ellipticEncryption.senderPublicString = function (senderPrivateKey) {
|
||
var senderKeyECData = {};
|
||
|
||
var curve = this.get_curve();
|
||
var G = this.get_G(curve);
|
||
var a = new BigInteger(senderPrivateKey);
|
||
var P = G.multiply(a);
|
||
senderKeyECData.XValuePublicString = P.getX().toBigInteger().toString();
|
||
senderKeyECData.YValuePublicString = P.getY().toBigInteger().toString();
|
||
|
||
return senderKeyECData;
|
||
};
|
||
|
||
//In real life ellipticEncryption.receiverPublicString is the public key of the receiver.
|
||
//you don't have to run receiverRandom and the bottom function
|
||
ellipticEncryption.receiverPublicString = function (receiverPublicKey) {
|
||
var receiverKeyECData = {};
|
||
|
||
var curve = this.get_curve();
|
||
var G = this.get_G(curve);
|
||
var a = new BigInteger(receiverPublicKey);
|
||
var P = G.multiply(a);
|
||
receiverKeyECData.XValuePublicString = P.getX()
|
||
.toBigInteger()
|
||
.toString();
|
||
receiverKeyECData.YValuePublicString = P.getY()
|
||
.toBigInteger()
|
||
.toString();
|
||
|
||
return receiverKeyECData;
|
||
};
|
||
|
||
ellipticEncryption.senderSharedKeyDerivation = function (
|
||
receiverPublicStringXValue,
|
||
receiverPublicStringYValue,
|
||
senderPrivateKey
|
||
) {
|
||
var senderDerivedKey = {};
|
||
var curve = this.get_curve();
|
||
var P = new EllipticCurve.PointFp(
|
||
curve,
|
||
curve.fromBigInteger(new BigInteger(receiverPublicStringXValue)),
|
||
curve.fromBigInteger(new BigInteger(receiverPublicStringYValue))
|
||
);
|
||
var a = new BigInteger(senderPrivateKey);
|
||
var S = P.multiply(a);
|
||
|
||
senderDerivedKey.XValue = S.getX().toBigInteger().toString();
|
||
senderDerivedKey.YValue = S.getY().toBigInteger().toString();
|
||
|
||
return senderDerivedKey;
|
||
};
|
||
|
||
ellipticEncryption.receiverSharedKeyDerivation = function (
|
||
senderPublicStringXValue,
|
||
senderPublicStringYValue,
|
||
receiverPrivateKey
|
||
) {
|
||
var receiverDerivedKey = {};
|
||
var curve = this.get_curve();
|
||
var P = new EllipticCurve.PointFp(
|
||
curve,
|
||
curve.fromBigInteger(new BigInteger(senderPublicStringXValue)),
|
||
curve.fromBigInteger(new BigInteger(senderPublicStringYValue))
|
||
);
|
||
var a = new BigInteger(receiverPrivateKey);
|
||
var S = P.multiply(a);
|
||
|
||
receiverDerivedKey.XValue = S.getX().toBigInteger().toString();
|
||
receiverDerivedKey.YValue = S.getY().toBigInteger().toString();
|
||
|
||
return receiverDerivedKey;
|
||
};
|
||
})("secp256k1");
|
||
|
||
// secrets.js - by Alexander Stetsyuk - released under MIT License
|
||
(function (exports, global) {
|
||
var defaults = {
|
||
bits: 8, // default number of bits
|
||
radix: 16, // work with HEX by default
|
||
minBits: 3,
|
||
maxBits: 20, // this permits 1,048,575 shares, though going this high is NOT recommended in JS!
|
||
|
||
bytesPerChar: 2,
|
||
maxBytesPerChar: 6, // Math.pow(256,7) > Math.pow(2,53)
|
||
|
||
// Primitive polynomials (in decimal form) for Galois Fields GF(2^n), for 2 <= n <= 30
|
||
// The index of each term in the array corresponds to the n for that polynomial
|
||
// i.e. to get the polynomial for n=16, use primitivePolynomials[16]
|
||
primitivePolynomials: [
|
||
null,
|
||
null,
|
||
1,
|
||
3,
|
||
3,
|
||
5,
|
||
3,
|
||
3,
|
||
29,
|
||
17,
|
||
9,
|
||
5,
|
||
83,
|
||
27,
|
||
43,
|
||
3,
|
||
45,
|
||
9,
|
||
39,
|
||
39,
|
||
9,
|
||
5,
|
||
3,
|
||
33,
|
||
27,
|
||
9,
|
||
71,
|
||
39,
|
||
9,
|
||
5,
|
||
83,
|
||
],
|
||
|
||
// warning for insecure PRNG
|
||
warning:
|
||
"WARNING:\nA secure random number generator was not found.\nUsing Math.random(), which is NOT cryptographically strong!",
|
||
};
|
||
|
||
// Protected settings object
|
||
var config = {};
|
||
|
||
/** @expose **/
|
||
exports.getConfig = function () {
|
||
return {
|
||
bits: config.bits,
|
||
unsafePRNG: config.unsafePRNG,
|
||
};
|
||
};
|
||
|
||
function init(bits) {
|
||
if (
|
||
bits &&
|
||
(typeof bits !== "number" ||
|
||
bits % 1 !== 0 ||
|
||
bits < defaults.minBits ||
|
||
bits > defaults.maxBits)
|
||
) {
|
||
throw new Error(
|
||
"Number of bits must be an integer between " +
|
||
defaults.minBits +
|
||
" and " +
|
||
defaults.maxBits +
|
||
", inclusive."
|
||
);
|
||
}
|
||
|
||
config.radix = defaults.radix;
|
||
config.bits = bits || defaults.bits;
|
||
config.size = Math.pow(2, config.bits);
|
||
config.max = config.size - 1;
|
||
|
||
// Construct the exp and log tables for multiplication.
|
||
var logs = [],
|
||
exps = [],
|
||
x = 1,
|
||
primitive = defaults.primitivePolynomials[config.bits];
|
||
for (var i = 0; i < config.size; i++) {
|
||
exps[i] = x;
|
||
logs[x] = i;
|
||
x <<= 1;
|
||
if (x >= config.size) {
|
||
x ^= primitive;
|
||
x &= config.max;
|
||
}
|
||
}
|
||
|
||
config.logs = logs;
|
||
config.exps = exps;
|
||
}
|
||
|
||
/** @expose **/
|
||
exports.init = init;
|
||
|
||
function isInited() {
|
||
if (
|
||
!config.bits ||
|
||
!config.size ||
|
||
!config.max ||
|
||
!config.logs ||
|
||
!config.exps ||
|
||
config.logs.length !== config.size ||
|
||
config.exps.length !== config.size
|
||
) {
|
||
return false;
|
||
}
|
||
return true;
|
||
}
|
||
|
||
// Returns a pseudo-random number generator of the form function(bits){}
|
||
// which should output a random string of 1's and 0's of length `bits`
|
||
function getRNG() {
|
||
var randomBits, crypto;
|
||
|
||
function construct(bits, arr, radix, size) {
|
||
var str = "",
|
||
i = 0,
|
||
len = arr.length - 1;
|
||
while (i < len || str.length < bits) {
|
||
str += padLeft(parseInt(arr[i], radix).toString(2), size);
|
||
i++;
|
||
}
|
||
str = str.substr(-bits);
|
||
if ((str.match(/0/g) || []).length === str.length) {
|
||
// all zeros?
|
||
return null;
|
||
} else {
|
||
return str;
|
||
}
|
||
}
|
||
|
||
// node.js crypto.randomBytes()
|
||
if (
|
||
typeof require === "function" &&
|
||
(crypto = require("crypto")) &&
|
||
(randomBits = crypto["randomBytes"])
|
||
) {
|
||
return function (bits) {
|
||
var bytes = Math.ceil(bits / 8),
|
||
str = null;
|
||
|
||
while (str === null) {
|
||
str = construct(bits, randomBits(bytes).toString("hex"), 16, 4);
|
||
}
|
||
return str;
|
||
};
|
||
}
|
||
|
||
// browsers with window.crypto.getRandomValues()
|
||
if (
|
||
global["crypto"] &&
|
||
typeof global["crypto"]["getRandomValues"] === "function" &&
|
||
typeof global["Uint32Array"] === "function"
|
||
) {
|
||
crypto = global["crypto"];
|
||
return function (bits) {
|
||
var elems = Math.ceil(bits / 32),
|
||
str = null,
|
||
arr = new global["Uint32Array"](elems);
|
||
|
||
while (str === null) {
|
||
crypto["getRandomValues"](arr);
|
||
str = construct(bits, arr, 10, 32);
|
||
}
|
||
|
||
return str;
|
||
};
|
||
}
|
||
|
||
// A totally insecure RNG!!! (except in Safari)
|
||
// Will produce a warning every time it is called.
|
||
config.unsafePRNG = true;
|
||
warn();
|
||
|
||
var bitsPerNum = 32;
|
||
var max = Math.pow(2, bitsPerNum) - 1;
|
||
return function (bits) {
|
||
var elems = Math.ceil(bits / bitsPerNum);
|
||
var arr = [],
|
||
str = null;
|
||
while (str === null) {
|
||
for (var i = 0; i < elems; i++) {
|
||
arr[i] = Math.floor(Math.random() * max + 1);
|
||
}
|
||
str = construct(bits, arr, 10, bitsPerNum);
|
||
}
|
||
return str;
|
||
};
|
||
}
|
||
|
||
// Warn about using insecure rng.
|
||
// Called when Math.random() is being used.
|
||
function warn() {
|
||
global["console"]["warn"](defaults.warning);
|
||
if (typeof global["alert"] === "function" && config.alert) {
|
||
global["alert"](defaults.warning);
|
||
}
|
||
}
|
||
|
||
// Set the PRNG to use. If no RNG function is supplied, pick a default using getRNG()
|
||
/** @expose **/
|
||
exports.setRNG = function (rng, alert) {
|
||
if (!isInited()) {
|
||
this.init();
|
||
}
|
||
config.unsafePRNG = false;
|
||
rng = rng || getRNG();
|
||
|
||
// test the RNG (5 times)
|
||
if (
|
||
typeof rng !== "function" ||
|
||
typeof rng(config.bits) !== "string" ||
|
||
!parseInt(rng(config.bits), 2) ||
|
||
rng(config.bits).length > config.bits ||
|
||
rng(config.bits).length < config.bits
|
||
) {
|
||
throw new Error(
|
||
"Random number generator is invalid. Supply an RNG of the form function(bits){} that returns a string containing 'bits' number of random 1's and 0's."
|
||
);
|
||
} else {
|
||
config.rng = rng;
|
||
}
|
||
config.alert = !!alert;
|
||
|
||
return !!config.unsafePRNG;
|
||
};
|
||
|
||
function isSetRNG() {
|
||
return typeof config.rng === "function";
|
||
}
|
||
|
||
// Generates a random bits-length number string using the PRNG
|
||
/** @expose **/
|
||
exports.random = function (bits) {
|
||
if (!isSetRNG()) {
|
||
this.setRNG();
|
||
}
|
||
|
||
if (typeof bits !== "number" || bits % 1 !== 0 || bits < 2) {
|
||
throw new Error("Number of bits must be an integer greater than 1.");
|
||
}
|
||
|
||
if (config.unsafePRNG) {
|
||
warn();
|
||
}
|
||
return bin2hex(config.rng(bits));
|
||
};
|
||
|
||
// Divides a `secret` number String str expressed in radix `inputRadix` (optional, default 16)
|
||
// into `numShares` shares, each expressed in radix `outputRadix` (optional, default to `inputRadix`),
|
||
// requiring `threshold` number of shares to reconstruct the secret.
|
||
// Optionally, zero-pads the secret to a length that is a multiple of padLength before sharing.
|
||
/** @expose **/
|
||
exports.share = function (
|
||
secret,
|
||
numShares,
|
||
threshold,
|
||
padLength,
|
||
withoutPrefix
|
||
) {
|
||
if (!isInited()) {
|
||
this.init();
|
||
}
|
||
if (!isSetRNG()) {
|
||
this.setRNG();
|
||
}
|
||
|
||
padLength = padLength || 0;
|
||
|
||
if (typeof secret !== "string") {
|
||
throw new Error("Secret must be a string.");
|
||
}
|
||
if (
|
||
typeof numShares !== "number" ||
|
||
numShares % 1 !== 0 ||
|
||
numShares < 2
|
||
) {
|
||
throw new Error(
|
||
"Number of shares must be an integer between 2 and 2^bits-1 (" +
|
||
config.max +
|
||
"), inclusive."
|
||
);
|
||
}
|
||
if (numShares > config.max) {
|
||
var neededBits = Math.ceil(Math.log(numShares + 1) / Math.LN2);
|
||
throw new Error(
|
||
"Number of shares must be an integer between 2 and 2^bits-1 (" +
|
||
config.max +
|
||
"), inclusive. To create " +
|
||
numShares +
|
||
" shares, use at least " +
|
||
neededBits +
|
||
" bits."
|
||
);
|
||
}
|
||
if (
|
||
typeof threshold !== "number" ||
|
||
threshold % 1 !== 0 ||
|
||
threshold < 2
|
||
) {
|
||
throw new Error(
|
||
"Threshold number of shares must be an integer between 2 and 2^bits-1 (" +
|
||
config.max +
|
||
"), inclusive."
|
||
);
|
||
}
|
||
if (threshold > config.max) {
|
||
var neededBits = Math.ceil(Math.log(threshold + 1) / Math.LN2);
|
||
throw new Error(
|
||
"Threshold number of shares must be an integer between 2 and 2^bits-1 (" +
|
||
config.max +
|
||
"), inclusive. To use a threshold of " +
|
||
threshold +
|
||
", use at least " +
|
||
neededBits +
|
||
" bits."
|
||
);
|
||
}
|
||
if (typeof padLength !== "number" || padLength % 1 !== 0) {
|
||
throw new Error("Zero-pad length must be an integer greater than 1.");
|
||
}
|
||
|
||
if (config.unsafePRNG) {
|
||
warn();
|
||
}
|
||
|
||
secret = "1" + hex2bin(secret); // append a 1 so that we can preserve the correct number of leading zeros in our secret
|
||
secret = split(secret, padLength);
|
||
var x = new Array(numShares),
|
||
y = new Array(numShares);
|
||
for (var i = 0, len = secret.length; i < len; i++) {
|
||
var subShares = this._getShares(secret[i], numShares, threshold);
|
||
for (var j = 0; j < numShares; j++) {
|
||
x[j] = x[j] || subShares[j].x.toString(config.radix);
|
||
y[j] = padLeft(subShares[j].y.toString(2)) + (y[j] ? y[j] : "");
|
||
}
|
||
}
|
||
var padding = config.max.toString(config.radix).length;
|
||
if (withoutPrefix) {
|
||
for (var i = 0; i < numShares; i++) {
|
||
x[i] = bin2hex(y[i]);
|
||
}
|
||
} else {
|
||
for (var i = 0; i < numShares; i++) {
|
||
x[i] =
|
||
config.bits.toString(36).toUpperCase() +
|
||
padLeft(x[i], padding) +
|
||
bin2hex(y[i]);
|
||
}
|
||
}
|
||
|
||
return x;
|
||
};
|
||
|
||
// This is the basic polynomial generation and evaluation function
|
||
// for a `config.bits`-length secret (NOT an arbitrary length)
|
||
// Note: no error-checking at this stage! If `secrets` is NOT
|
||
// a NUMBER less than 2^bits-1, the output will be incorrect!
|
||
/** @expose **/
|
||
exports._getShares = function (secret, numShares, threshold) {
|
||
var shares = [];
|
||
var coeffs = [secret];
|
||
|
||
for (var i = 1; i < threshold; i++) {
|
||
coeffs[i] = parseInt(config.rng(config.bits), 2);
|
||
}
|
||
for (var i = 1, len = numShares + 1; i < len; i++) {
|
||
shares[i - 1] = {
|
||
x: i,
|
||
y: horner(i, coeffs),
|
||
};
|
||
}
|
||
return shares;
|
||
};
|
||
|
||
// Polynomial evaluation at `x` using Horner's Method
|
||
// TODO: this can possibly be sped up using other methods
|
||
// NOTE: fx=fx * x + coeff[i] -> exp(log(fx) + log(x)) + coeff[i],
|
||
// so if fx===0, just set fx to coeff[i] because
|
||
// using the exp/log form will result in incorrect value
|
||
function horner(x, coeffs) {
|
||
var logx = config.logs[x];
|
||
var fx = 0;
|
||
for (var i = coeffs.length - 1; i >= 0; i--) {
|
||
if (fx === 0) {
|
||
fx = coeffs[i];
|
||
continue;
|
||
}
|
||
fx = config.exps[(logx + config.logs[fx]) % config.max] ^ coeffs[i];
|
||
}
|
||
return fx;
|
||
}
|
||
|
||
function inArray(arr, val) {
|
||
for (var i = 0, len = arr.length; i < len; i++) {
|
||
if (arr[i] === val) {
|
||
return true;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
function processShare(share) {
|
||
var bits = parseInt(share[0], 36);
|
||
if (
|
||
bits &&
|
||
(typeof bits !== "number" ||
|
||
bits % 1 !== 0 ||
|
||
bits < defaults.minBits ||
|
||
bits > defaults.maxBits)
|
||
) {
|
||
throw new Error(
|
||
"Number of bits must be an integer between " +
|
||
defaults.minBits +
|
||
" and " +
|
||
defaults.maxBits +
|
||
", inclusive."
|
||
);
|
||
}
|
||
|
||
var max = Math.pow(2, bits) - 1;
|
||
var idLength = max.toString(config.radix).length;
|
||
|
||
var id = parseInt(share.substr(1, idLength), config.radix);
|
||
if (typeof id !== "number" || id % 1 !== 0 || id < 1 || id > max) {
|
||
throw new Error(
|
||
"Share id must be an integer between 1 and " +
|
||
config.max +
|
||
", inclusive."
|
||
);
|
||
}
|
||
share = share.substr(idLength + 1);
|
||
if (!share.length) {
|
||
throw new Error("Invalid share: zero-length share.");
|
||
}
|
||
return {
|
||
bits: bits,
|
||
id: id,
|
||
value: share,
|
||
};
|
||
}
|
||
|
||
/** @expose **/
|
||
exports._processShare = processShare;
|
||
|
||
// Protected method that evaluates the Lagrange interpolation
|
||
// polynomial at x=`at` for individual config.bits-length
|
||
// segments of each share in the `shares` Array.
|
||
// Each share is expressed in base `inputRadix`. The output
|
||
// is expressed in base `outputRadix'
|
||
function combine(at, shares) {
|
||
var setBits,
|
||
share,
|
||
x = [],
|
||
y = [],
|
||
result = "",
|
||
idx;
|
||
|
||
for (var i = 0, len = shares.length; i < len; i++) {
|
||
share = processShare(shares[i]);
|
||
if (typeof setBits === "undefined") {
|
||
setBits = share["bits"];
|
||
} else if (share["bits"] !== setBits) {
|
||
throw new Error("Mismatched shares: Different bit settings.");
|
||
}
|
||
|
||
if (config.bits !== setBits) {
|
||
init(setBits);
|
||
}
|
||
|
||
if (inArray(x, share["id"])) {
|
||
// repeated x value?
|
||
continue;
|
||
}
|
||
|
||
idx = x.push(share["id"]) - 1;
|
||
share = split(hex2bin(share["value"]));
|
||
for (var j = 0, len2 = share.length; j < len2; j++) {
|
||
y[j] = y[j] || [];
|
||
y[j][idx] = share[j];
|
||
}
|
||
}
|
||
|
||
for (var i = 0, len = y.length; i < len; i++) {
|
||
result = padLeft(lagrange(at, x, y[i]).toString(2)) + result;
|
||
}
|
||
|
||
if (at === 0) {
|
||
// reconstructing the secret
|
||
var idx = result.indexOf("1"); //find the first 1
|
||
return bin2hex(result.slice(idx + 1));
|
||
} else {
|
||
// generating a new share
|
||
return bin2hex(result);
|
||
}
|
||
}
|
||
|
||
// Combine `shares` Array into the original secret
|
||
/** @expose **/
|
||
exports.combine = function (shares) {
|
||
return combine(0, shares);
|
||
};
|
||
|
||
// Generate a new share with id `id` (a number between 1 and 2^bits-1)
|
||
// `id` can be a Number or a String in the default radix (16)
|
||
/** @expose **/
|
||
exports.newShare = function (id, shares) {
|
||
if (typeof id === "string") {
|
||
id = parseInt(id, config.radix);
|
||
}
|
||
|
||
var share = processShare(shares[0]);
|
||
var max = Math.pow(2, share["bits"]) - 1;
|
||
|
||
if (typeof id !== "number" || id % 1 !== 0 || id < 1 || id > max) {
|
||
throw new Error(
|
||
"Share id must be an integer between 1 and " +
|
||
config.max +
|
||
", inclusive."
|
||
);
|
||
}
|
||
|
||
var padding = max.toString(config.radix).length;
|
||
return (
|
||
config.bits.toString(36).toUpperCase() +
|
||
padLeft(id.toString(config.radix), padding) +
|
||
combine(id, shares)
|
||
);
|
||
};
|
||
|
||
// Evaluate the Lagrange interpolation polynomial at x = `at`
|
||
// using x and y Arrays that are of the same length, with
|
||
// corresponding elements constituting points on the polynomial.
|
||
function lagrange(at, x, y) {
|
||
var sum = 0,
|
||
product,
|
||
i,
|
||
j;
|
||
|
||
for (var i = 0, len = x.length; i < len; i++) {
|
||
if (!y[i]) {
|
||
continue;
|
||
}
|
||
|
||
product = config.logs[y[i]];
|
||
for (var j = 0; j < len; j++) {
|
||
if (i === j) {
|
||
continue;
|
||
}
|
||
if (at === x[j]) {
|
||
// happens when computing a share that is in the list of shares used to compute it
|
||
product = -1; // fix for a zero product term, after which the sum should be sum^0 = sum, not sum^1
|
||
break;
|
||
}
|
||
product =
|
||
(product +
|
||
config.logs[at ^ x[j]] -
|
||
config.logs[x[i] ^ x[j]] +
|
||
config.max) /* to make sure it's not negative */ %
|
||
config.max;
|
||
}
|
||
|
||
sum = product === -1 ? sum : sum ^ config.exps[product]; // though exps[-1]= undefined and undefined ^ anything = anything in chrome, this behavior may not hold everywhere, so do the check
|
||
}
|
||
return sum;
|
||
}
|
||
|
||
/** @expose **/
|
||
exports._lagrange = lagrange;
|
||
|
||
// Splits a number string `bits`-length segments, after first
|
||
// optionally zero-padding it to a length that is a multiple of `padLength.
|
||
// Returns array of integers (each less than 2^bits-1), with each element
|
||
// representing a `bits`-length segment of the input string from right to left,
|
||
// i.e. parts[0] represents the right-most `bits`-length segment of the input string.
|
||
function split(str, padLength) {
|
||
if (padLength) {
|
||
str = padLeft(str, padLength);
|
||
}
|
||
var parts = [];
|
||
for (var i = str.length; i > config.bits; i -= config.bits) {
|
||
parts.push(parseInt(str.slice(i - config.bits, i), 2));
|
||
}
|
||
parts.push(parseInt(str.slice(0, i), 2));
|
||
return parts;
|
||
}
|
||
|
||
// Pads a string `str` with zeros on the left so that its length is a multiple of `bits`
|
||
function padLeft(str, bits) {
|
||
bits = bits || config.bits;
|
||
var missing = str.length % bits;
|
||
return (missing ? new Array(bits - missing + 1).join("0") : "") + str;
|
||
}
|
||
|
||
function hex2bin(str) {
|
||
var bin = "",
|
||
num;
|
||
for (var i = str.length - 1; i >= 0; i--) {
|
||
num = parseInt(str[i], 16);
|
||
if (isNaN(num)) {
|
||
throw new Error("Invalid hex character.");
|
||
}
|
||
bin = padLeft(num.toString(2), 4) + bin;
|
||
}
|
||
return bin;
|
||
}
|
||
|
||
function bin2hex(str) {
|
||
var hex = "",
|
||
num;
|
||
str = padLeft(str, 4);
|
||
for (var i = str.length; i >= 4; i -= 4) {
|
||
num = parseInt(str.slice(i - 4, i), 2);
|
||
if (isNaN(num)) {
|
||
throw new Error("Invalid binary character.");
|
||
}
|
||
hex = num.toString(16) + hex;
|
||
}
|
||
return hex;
|
||
}
|
||
|
||
// Converts a given UTF16 character string to the HEX representation.
|
||
// Each character of the input string is represented by
|
||
// `bytesPerChar` bytes in the output string.
|
||
/** @expose **/
|
||
exports.str2hex = function (str, bytesPerChar) {
|
||
if (typeof str !== "string") {
|
||
throw new Error("Input must be a character string.");
|
||
}
|
||
bytesPerChar = bytesPerChar || defaults.bytesPerChar;
|
||
|
||
if (
|
||
typeof bytesPerChar !== "number" ||
|
||
bytesPerChar % 1 !== 0 ||
|
||
bytesPerChar < 1 ||
|
||
bytesPerChar > defaults.maxBytesPerChar
|
||
) {
|
||
throw new Error(
|
||
"Bytes per character must be an integer between 1 and " +
|
||
defaults.maxBytesPerChar +
|
||
", inclusive."
|
||
);
|
||
}
|
||
|
||
var hexChars = 2 * bytesPerChar;
|
||
var max = Math.pow(16, hexChars) - 1;
|
||
var out = "",
|
||
num;
|
||
for (var i = 0, len = str.length; i < len; i++) {
|
||
num = str[i].charCodeAt();
|
||
if (isNaN(num)) {
|
||
throw new Error("Invalid character: " + str[i]);
|
||
} else if (num > max) {
|
||
var neededBytes = Math.ceil(Math.log(num + 1) / Math.log(256));
|
||
throw new Error(
|
||
"Invalid character code (" +
|
||
num +
|
||
"). Maximum allowable is 256^bytes-1 (" +
|
||
max +
|
||
"). To convert this character, use at least " +
|
||
neededBytes +
|
||
" bytes."
|
||
);
|
||
} else {
|
||
out = padLeft(num.toString(16), hexChars) + out;
|
||
}
|
||
}
|
||
return out;
|
||
};
|
||
|
||
// Converts a given HEX number string to a UTF16 character string.
|
||
/** @expose **/
|
||
exports.hex2str = function (str, bytesPerChar) {
|
||
if (typeof str !== "string") {
|
||
throw new Error("Input must be a hexadecimal string.");
|
||
}
|
||
bytesPerChar = bytesPerChar || defaults.bytesPerChar;
|
||
|
||
if (
|
||
typeof bytesPerChar !== "number" ||
|
||
bytesPerChar % 1 !== 0 ||
|
||
bytesPerChar < 1 ||
|
||
bytesPerChar > defaults.maxBytesPerChar
|
||
) {
|
||
throw new Error(
|
||
"Bytes per character must be an integer between 1 and " +
|
||
defaults.maxBytesPerChar +
|
||
", inclusive."
|
||
);
|
||
}
|
||
|
||
var hexChars = 2 * bytesPerChar;
|
||
var out = "";
|
||
str = padLeft(str, hexChars);
|
||
for (var i = 0, len = str.length; i < len; i += hexChars) {
|
||
out =
|
||
String.fromCharCode(parseInt(str.slice(i, i + hexChars), 16)) + out;
|
||
}
|
||
return out;
|
||
};
|
||
|
||
// by default, initialize without an RNG
|
||
exports.init();
|
||
})(
|
||
typeof module !== "undefined" && module["exports"]
|
||
? module["exports"]
|
||
: (window["shamirSecretShare"] = {}),
|
||
typeof global !== "undefined" ? global : window
|
||
);
|
||
|
||
//For diff base
|
||
/*
|
||
Functions available:
|
||
diff(originalObj, updatedObj) returns the difference of the original and updated objects
|
||
addedDiff(original, updatedObj) returns only the values added to the updated object
|
||
deletedDiff(original, updatedObj) returns only the values deleted in the updated object
|
||
updatedDiff(original, updatedObj) returns only the values that have been changed in the updated object
|
||
findDifference(original, updatedObj) returns an object with the added, deleted and updated differences
|
||
mergeRecurcive(original, diff) this will get you a new object that will merge all the changes between the old object and the new object
|
||
*/
|
||
(function () {
|
||
const isDate = (d) => d instanceof Date;
|
||
const isEmpty = (o) => Object.keys(o).length === 0;
|
||
const isObject = (o) => o != null && typeof o === "object";
|
||
const properObject = (o) =>
|
||
isObject(o) && !o.hasOwnProperty
|
||
? {
|
||
...o,
|
||
}
|
||
: o;
|
||
|
||
const getLargerArray = (l, r) => (l.length > r.length ? l : r);
|
||
|
||
const preserve = (diff, left, right) => {
|
||
if (!isObject(diff)) return diff;
|
||
|
||
return Object.keys(diff).reduce((acc, key) => {
|
||
const leftArray = left[key];
|
||
const rightArray = right[key];
|
||
|
||
if (Array.isArray(leftArray) && Array.isArray(rightArray)) {
|
||
const array = [...getLargerArray(leftArray, rightArray)];
|
||
return {
|
||
...acc,
|
||
[key]: array.reduce((acc2, item, index) => {
|
||
if (diff[key].hasOwnProperty(index)) {
|
||
acc2[index] = preserve(
|
||
diff[key][index],
|
||
leftArray[index],
|
||
rightArray[index]
|
||
); // diff recurse and check for nested arrays
|
||
return acc2;
|
||
}
|
||
|
||
delete acc2[index]; // no diff aka empty
|
||
return acc2;
|
||
}, array),
|
||
};
|
||
}
|
||
|
||
return {
|
||
...acc,
|
||
[key]: diff[key],
|
||
};
|
||
}, {});
|
||
};
|
||
|
||
const updatedDiff = (lhs, rhs) => {
|
||
if (lhs === rhs) return {};
|
||
|
||
if (!isObject(lhs) || !isObject(rhs)) return rhs;
|
||
|
||
const l = properObject(lhs);
|
||
const r = properObject(rhs);
|
||
|
||
if (isDate(l) || isDate(r)) {
|
||
if (l.valueOf() == r.valueOf()) return {};
|
||
return r;
|
||
}
|
||
|
||
return Object.keys(r).reduce((acc, key) => {
|
||
if (l.hasOwnProperty(key)) {
|
||
const difference = updatedDiff(l[key], r[key]);
|
||
|
||
if (
|
||
isObject(difference) &&
|
||
isEmpty(difference) &&
|
||
!isDate(difference)
|
||
)
|
||
return acc;
|
||
|
||
return {
|
||
...acc,
|
||
[key]: difference,
|
||
};
|
||
}
|
||
|
||
return acc;
|
||
}, {});
|
||
};
|
||
|
||
const diff = (lhs, rhs) => {
|
||
if (lhs === rhs) return {}; // equal return no diff
|
||
|
||
if (!isObject(lhs) || !isObject(rhs)) return rhs; // return updated rhs
|
||
|
||
const l = properObject(lhs);
|
||
const r = properObject(rhs);
|
||
|
||
const deletedValues = Object.keys(l).reduce((acc, key) => {
|
||
return r.hasOwnProperty(key)
|
||
? acc
|
||
: {
|
||
...acc,
|
||
[key]: null,
|
||
};
|
||
}, {});
|
||
|
||
if (isDate(l) || isDate(r)) {
|
||
if (l.valueOf() == r.valueOf()) return {};
|
||
return r;
|
||
}
|
||
|
||
return Object.keys(r).reduce((acc, key) => {
|
||
if (!l.hasOwnProperty(key))
|
||
return {
|
||
...acc,
|
||
[key]: r[key],
|
||
}; // return added r key
|
||
|
||
const difference = diff(l[key], r[key]);
|
||
|
||
if (
|
||
isObject(difference) &&
|
||
isEmpty(difference) &&
|
||
!isDate(difference)
|
||
)
|
||
return acc; // return no diff
|
||
|
||
return {
|
||
...acc,
|
||
[key]: difference,
|
||
}; // return updated key
|
||
}, deletedValues);
|
||
};
|
||
|
||
const addedDiff = (lhs, rhs) => {
|
||
if (lhs === rhs || !isObject(lhs) || !isObject(rhs)) return {};
|
||
|
||
const l = properObject(lhs);
|
||
const r = properObject(rhs);
|
||
|
||
return Object.keys(r).reduce((acc, key) => {
|
||
if (l.hasOwnProperty(key)) {
|
||
const difference = addedDiff(l[key], r[key]);
|
||
|
||
if (isObject(difference) && isEmpty(difference)) return acc;
|
||
|
||
return {
|
||
...acc,
|
||
[key]: difference,
|
||
};
|
||
}
|
||
|
||
return {
|
||
...acc,
|
||
[key]: r[key],
|
||
};
|
||
}, {});
|
||
};
|
||
|
||
const arrayDiff = (lhs, rhs) => {
|
||
if (lhs === rhs) return {}; // equal return no diff
|
||
|
||
if (!isObject(lhs) || !isObject(rhs)) return rhs; // return updated rhs
|
||
|
||
const l = properObject(lhs);
|
||
const r = properObject(rhs);
|
||
|
||
const deletedValues = Object.keys(l).reduce((acc, key) => {
|
||
return r.hasOwnProperty(key)
|
||
? acc
|
||
: {
|
||
...acc,
|
||
[key]: null,
|
||
};
|
||
}, {});
|
||
|
||
if (isDate(l) || isDate(r)) {
|
||
if (l.valueOf() == r.valueOf()) return {};
|
||
return r;
|
||
}
|
||
|
||
if (Array.isArray(r) && Array.isArray(l)) {
|
||
const deletedValues = l.reduce((acc, item, index) => {
|
||
return r.hasOwnProperty(index)
|
||
? acc.concat(item)
|
||
: acc.concat(null);
|
||
}, []);
|
||
|
||
return r.reduce((acc, rightItem, index) => {
|
||
if (!deletedValues.hasOwnProperty(index)) {
|
||
return acc.concat(rightItem);
|
||
}
|
||
|
||
const leftItem = l[index];
|
||
const difference = diff(rightItem, leftItem);
|
||
|
||
if (
|
||
isObject(difference) &&
|
||
isEmpty(difference) &&
|
||
!isDate(difference)
|
||
) {
|
||
delete acc[index];
|
||
return acc; // return no diff
|
||
}
|
||
|
||
return acc
|
||
.slice(0, index)
|
||
.concat(rightItem)
|
||
.concat(acc.slice(index + 1)); // return updated key
|
||
}, deletedValues);
|
||
}
|
||
|
||
return Object.keys(r).reduce((acc, key) => {
|
||
if (!l.hasOwnProperty(key))
|
||
return {
|
||
...acc,
|
||
[key]: r[key],
|
||
}; // return added r key
|
||
|
||
const difference = diff(l[key], r[key]);
|
||
|
||
if (
|
||
isObject(difference) &&
|
||
isEmpty(difference) &&
|
||
!isDate(difference)
|
||
)
|
||
return acc; // return no diff
|
||
|
||
return {
|
||
...acc,
|
||
[key]: difference,
|
||
}; // return updated key
|
||
}, deletedValues);
|
||
};
|
||
|
||
const deletedDiff = (lhs, rhs) => {
|
||
if (lhs === rhs || !isObject(lhs) || !isObject(rhs)) return {};
|
||
|
||
const l = properObject(lhs);
|
||
const r = properObject(rhs);
|
||
|
||
return Object.keys(l).reduce((acc, key) => {
|
||
if (r.hasOwnProperty(key)) {
|
||
const difference = deletedDiff(l[key], r[key]);
|
||
|
||
if (isObject(difference) && isEmpty(difference)) return acc;
|
||
|
||
return {
|
||
...acc,
|
||
[key]: difference,
|
||
};
|
||
}
|
||
|
||
return {
|
||
...acc,
|
||
[key]: null,
|
||
};
|
||
}, {});
|
||
};
|
||
|
||
window.findDifference = (lhs, rhs) => ({
|
||
added: addedDiff(lhs, rhs),
|
||
deleted: deletedDiff(lhs, rhs),
|
||
updated: updatedDiff(lhs, rhs),
|
||
});
|
||
|
||
const mergeRecursive = (obj1, obj2) => {
|
||
for (var p in obj2) {
|
||
try {
|
||
if (obj2[p].constructor == Object) {
|
||
obj1[p] = mergeRecursive(obj1[p], obj2[p]);
|
||
}
|
||
// Property in destination object set; update its value.
|
||
else if (Ext.isArray(obj2[p])) {
|
||
// obj1[p] = [];
|
||
if (obj2[p].length < 1) {
|
||
obj1[p] = obj2[p];
|
||
} else {
|
||
obj1[p] = mergeRecursive(obj1[p], obj2[p]);
|
||
}
|
||
} else {
|
||
obj1[p] = obj2[p];
|
||
}
|
||
} catch (e) {
|
||
// Property in destination object not set; create it and set its value.
|
||
obj1[p] = obj2[p];
|
||
}
|
||
}
|
||
return obj1;
|
||
};
|
||
|
||
/*
|
||
var test = {
|
||
foo : {
|
||
bar : {
|
||
baz : null
|
||
}
|
||
},
|
||
bar : 1
|
||
};
|
||
cleanse(test);
|
||
|
||
Rohit: Added a small fix for object being entered as Array*/
|
||
|
||
const cleanse = (obj) => {
|
||
Object.keys(obj).forEach((key) => {
|
||
var value = obj[key];
|
||
if (typeof value === "object" && value !== null) {
|
||
// Recurse...
|
||
cleanse(value);
|
||
// ...and remove if now "empty" (NOTE: insert your definition of "empty" here)
|
||
if (!Object.keys(value).length) {
|
||
delete obj[key];
|
||
}
|
||
} else if (value === null) {
|
||
// null, remove it
|
||
delete obj[key];
|
||
}
|
||
});
|
||
|
||
if (obj.constructor.toString().indexOf("Array") != -1) {
|
||
obj = obj.filter(function (el) {
|
||
return el != null;
|
||
});
|
||
}
|
||
|
||
return obj;
|
||
};
|
||
|
||
/*obj is original object or array, diff is the output of findDifference */
|
||
window.mergeDifference = (obj, diff) => {
|
||
if (Object.keys(diff.updated).length !== 0)
|
||
obj = mergeRecursive(obj, diff.updated);
|
||
if (Object.keys(diff.deleted).length !== 0) {
|
||
obj = mergeRecursive(obj, diff.deleted);
|
||
obj = cleanse(obj);
|
||
}
|
||
if (Object.keys(diff.added).length !== 0)
|
||
obj = mergeRecursive(obj, diff.added);
|
||
return obj;
|
||
};
|
||
})();
|
||
</script>
|
||
|
||
<script id="floCrypto">
|
||
/* FLO Crypto Operators*/
|
||
const floCrypto = {
|
||
util: {
|
||
p: BigInteger(
|
||
"FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFC2F",
|
||
16
|
||
),
|
||
|
||
ecparams: EllipticCurve.getSECCurveByName("secp256k1"),
|
||
|
||
exponent1: function () {
|
||
return this.p.add(BigInteger.ONE).divide(BigInteger("4"));
|
||
},
|
||
|
||
calculateY: function (x) {
|
||
let p = this.p;
|
||
let exp = this.exponent1();
|
||
// x is x value of public key in BigInteger format without 02 or 03 or 04 prefix
|
||
return x
|
||
.modPow(BigInteger("3"), p)
|
||
.add(BigInteger("7"))
|
||
.mod(p)
|
||
.modPow(exp, p);
|
||
},
|
||
getUncompressedPublicKey: function (compressedPublicKey) {
|
||
const p = this.p;
|
||
// Fetch x from compressedPublicKey
|
||
let pubKeyBytes = Crypto.util.hexToBytes(compressedPublicKey);
|
||
const prefix = pubKeyBytes.shift(); // remove prefix
|
||
let prefix_modulus = prefix % 2;
|
||
pubKeyBytes.unshift(0); // add prefix 0
|
||
let x = new BigInteger(pubKeyBytes);
|
||
let xDecimalValue = x.toString();
|
||
// Fetch y
|
||
let y = this.calculateY(x);
|
||
let yDecimalValue = y.toString();
|
||
// verify y value
|
||
let resultBigInt = y.mod(BigInteger("2"));
|
||
let check = resultBigInt.toString() % 2;
|
||
if (prefix_modulus !== check) {
|
||
yDecimalValue = y.negate().mod(p).toString();
|
||
}
|
||
return {
|
||
x: xDecimalValue,
|
||
y: yDecimalValue,
|
||
};
|
||
},
|
||
|
||
getSenderPublicKeyString: function () {
|
||
privateKey = ellipticCurveEncryption.senderRandom();
|
||
senderPublicKeyString =
|
||
ellipticCurveEncryption.senderPublicString(privateKey);
|
||
return {
|
||
privateKey: privateKey,
|
||
senderPublicKeyString: senderPublicKeyString,
|
||
};
|
||
},
|
||
|
||
deriveSharedKeySender: function (
|
||
receiverCompressedPublicKey,
|
||
senderPrivateKey
|
||
) {
|
||
try {
|
||
let receiverPublicKeyString = this.getUncompressedPublicKey(
|
||
receiverCompressedPublicKey
|
||
);
|
||
var senderDerivedKey =
|
||
ellipticCurveEncryption.senderSharedKeyDerivation(
|
||
receiverPublicKeyString.x,
|
||
receiverPublicKeyString.y,
|
||
senderPrivateKey
|
||
);
|
||
return senderDerivedKey;
|
||
} catch (error) {
|
||
return new Error(error);
|
||
}
|
||
},
|
||
|
||
deriveReceiverSharedKey: function (
|
||
senderPublicKeyString,
|
||
receiverPrivateKey
|
||
) {
|
||
return ellipticCurveEncryption.receiverSharedKeyDerivation(
|
||
senderPublicKeyString.XValuePublicString,
|
||
senderPublicKeyString.YValuePublicString,
|
||
receiverPrivateKey
|
||
);
|
||
},
|
||
|
||
getReceiverPublicKeyString: function (privateKey) {
|
||
return ellipticCurveEncryption.receiverPublicString(privateKey);
|
||
},
|
||
|
||
deriveSharedKeyReceiver: function (
|
||
senderPublicKeyString,
|
||
receiverPrivateKey
|
||
) {
|
||
try {
|
||
return ellipticCurveEncryption.receiverSharedKeyDerivation(
|
||
senderPublicKeyString.XValuePublicString,
|
||
senderPublicKeyString.YValuePublicString,
|
||
receiverPrivateKey
|
||
);
|
||
} catch (error) {
|
||
return new Error(error);
|
||
}
|
||
},
|
||
|
||
wifToDecimal: function (pk_wif, isPubKeyCompressed = false) {
|
||
let pk = Bitcoin.Base58.decode(pk_wif);
|
||
pk.shift();
|
||
pk.splice(-4, 4);
|
||
//If the private key corresponded to a compressed public key, also drop the last byte (it should be 0x01).
|
||
if (isPubKeyCompressed == true) pk.pop();
|
||
pk.unshift(0);
|
||
privateKeyDecimal = BigInteger(pk).toString();
|
||
privateKeyHex = Crypto.util.bytesToHex(pk);
|
||
return {
|
||
privateKeyDecimal: privateKeyDecimal,
|
||
privateKeyHex: privateKeyHex,
|
||
};
|
||
},
|
||
},
|
||
|
||
//generate a random Interger within range
|
||
randInt: function (min, max) {
|
||
min = Math.ceil(min);
|
||
max = Math.floor(max);
|
||
return Math.floor(Math.random() * (max - min + 1)) + min;
|
||
},
|
||
|
||
//generate a random String within length (options : alphaNumeric chars only)
|
||
randString: function (length, alphaNumeric = false) {
|
||
var result = "";
|
||
if (alphaNumeric)
|
||
var characters =
|
||
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
|
||
else
|
||
var characters =
|
||
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_+-./*?@#&$<>=[]{}():";
|
||
for (var i = 0; i < length; i++)
|
||
result += characters.charAt(
|
||
Math.floor(Math.random() * characters.length)
|
||
);
|
||
return result;
|
||
},
|
||
|
||
//Encrypt Data using public-key
|
||
encryptData: function (data, receiverCompressedPublicKey) {
|
||
var senderECKeyData = this.util.getSenderPublicKeyString();
|
||
var senderDerivedKey = this.util.deriveSharedKeySender(
|
||
receiverCompressedPublicKey,
|
||
senderECKeyData.privateKey
|
||
);
|
||
let senderKey = senderDerivedKey.XValue + senderDerivedKey.YValue;
|
||
let secret = Crypto.AES.encrypt(data, senderKey);
|
||
return {
|
||
secret: secret,
|
||
senderPublicKeyString: senderECKeyData.senderPublicKeyString,
|
||
};
|
||
},
|
||
|
||
//Decrypt Data using private-key
|
||
decryptData: function (data, myPrivateKey) {
|
||
var receiverECKeyData = {};
|
||
if (typeof myPrivateKey !== "string")
|
||
throw new Error("No private key found.");
|
||
|
||
let privateKey = this.util.wifToDecimal(myPrivateKey, true);
|
||
if (typeof privateKey.privateKeyDecimal !== "string")
|
||
throw new Error("Failed to detremine your private key.");
|
||
receiverECKeyData.privateKey = privateKey.privateKeyDecimal;
|
||
|
||
var receiverDerivedKey = this.util.deriveReceiverSharedKey(
|
||
data.senderPublicKeyString,
|
||
receiverECKeyData.privateKey
|
||
);
|
||
|
||
let receiverKey = receiverDerivedKey.XValue + receiverDerivedKey.YValue;
|
||
let decryptMsg = Crypto.AES.decrypt(data.secret, receiverKey);
|
||
return decryptMsg;
|
||
},
|
||
|
||
//Sign data using private-key
|
||
signData: function (data, privateKeyHex) {
|
||
var key = new Bitcoin.ECKey(privateKeyHex);
|
||
key.setCompressed(true);
|
||
|
||
var privateKeyArr = key.getBitcoinPrivateKeyByteArray();
|
||
privateKey = BigInteger.fromByteArrayUnsigned(privateKeyArr);
|
||
var messageHash = Crypto.SHA256(data);
|
||
|
||
var messageHashBigInteger = new BigInteger(messageHash);
|
||
var messageSign = Bitcoin.ECDSA.sign(messageHashBigInteger, key.priv);
|
||
|
||
var sighex = Crypto.util.bytesToHex(messageSign);
|
||
return sighex;
|
||
},
|
||
|
||
//Verify signatue of the data using public-key
|
||
verifySign: function (data, signatureHex, publicKeyHex) {
|
||
var msgHash = Crypto.SHA256(data);
|
||
var messageHashBigInteger = new BigInteger(msgHash);
|
||
|
||
var sigBytes = Crypto.util.hexToBytes(signatureHex);
|
||
var signature = Bitcoin.ECDSA.parseSig(sigBytes);
|
||
|
||
var publicKeyPoint = this.util.ecparams
|
||
.getCurve()
|
||
.decodePointHex(publicKeyHex);
|
||
|
||
var verify = Bitcoin.ECDSA.verifyRaw(
|
||
messageHashBigInteger,
|
||
signature.r,
|
||
signature.s,
|
||
publicKeyPoint
|
||
);
|
||
return verify;
|
||
},
|
||
|
||
//Generates a new flo ID and returns private-key, public-key and floID
|
||
generateNewID: function () {
|
||
try {
|
||
var key = new Bitcoin.ECKey(false);
|
||
key.setCompressed(true);
|
||
return {
|
||
floID: key.getBitcoinAddress(),
|
||
pubKey: key.getPubKeyHex(),
|
||
privKey: key.getBitcoinWalletImportFormat(),
|
||
};
|
||
} catch (e) {
|
||
console.error(e);
|
||
}
|
||
},
|
||
|
||
//Returns public-key from private-key
|
||
getPubKeyHex: function (privateKeyHex) {
|
||
if (!privateKeyHex) return null;
|
||
var key = new Bitcoin.ECKey(privateKeyHex);
|
||
if (key.priv == null) return null;
|
||
key.setCompressed(true);
|
||
var pubkeyHex = key.getPubKeyHex();
|
||
return pubkeyHex;
|
||
},
|
||
|
||
//Returns flo-ID from public-key
|
||
getFloIDfromPubkeyHex: function (pubkeyHex) {
|
||
try {
|
||
var key = new Bitcoin.ECKey().setPub(pubkeyHex);
|
||
var floID = key.getBitcoinAddress();
|
||
return floID;
|
||
} catch (e) {
|
||
console.error(e);
|
||
}
|
||
},
|
||
|
||
//Verify the private-key for the given public-key or flo-ID
|
||
verifyPrivKey: function (privateKeyHex, pubKey_floID, isfloID = true) {
|
||
try {
|
||
var key = new Bitcoin.ECKey(privateKeyHex);
|
||
if (key.priv == null) return false;
|
||
key.setCompressed(true);
|
||
if (isfloID && pubKey_floID == key.getBitcoinAddress()) return true;
|
||
else if (!isfloID && pubKey_floID == key.getPubKeyHex()) return true;
|
||
else return false;
|
||
} catch (e) {
|
||
console.error(e);
|
||
}
|
||
},
|
||
|
||
//Check if the given Address is valid or not
|
||
validateAddr: function (inpAddr) {
|
||
try {
|
||
var addr = new Bitcoin.Address(inpAddr);
|
||
return true;
|
||
} catch {
|
||
return false;
|
||
}
|
||
},
|
||
|
||
//Split the str using shamir's Secret and Returns the shares
|
||
createShamirsSecretShares: function (str, total_shares, threshold_limit) {
|
||
try {
|
||
if (str.length > 0) {
|
||
var strHex = shamirSecretShare.str2hex(str);
|
||
var shares = shamirSecretShare.share(
|
||
strHex,
|
||
total_shares,
|
||
threshold_limit
|
||
);
|
||
return shares;
|
||
}
|
||
return false;
|
||
} catch {
|
||
return false;
|
||
}
|
||
},
|
||
|
||
//Verifies the shares and str
|
||
verifyShamirsSecret: function (sharesArray, str) {
|
||
return str && this.retrieveShamirSecret(sharesArray) === str;
|
||
},
|
||
|
||
//Returns the retrived secret by combining the shamirs shares
|
||
retrieveShamirSecret: function (sharesArray) {
|
||
try {
|
||
if (sharesArray.length > 0) {
|
||
var comb = shamirSecretShare.combine(
|
||
sharesArray.slice(0, sharesArray.length)
|
||
);
|
||
comb = shamirSecretShare.hex2str(comb);
|
||
return comb;
|
||
}
|
||
return false;
|
||
} catch {
|
||
return false;
|
||
}
|
||
},
|
||
};
|
||
</script>
|
||
|
||
<script id="floBlockchainAPI">
|
||
/* FLO Blockchain Operator to send/receive data from blockchain using API calls*/
|
||
const floBlockchainAPI = {
|
||
util: {
|
||
serverList: floGlobals.apiURL[floGlobals.blockchain].slice(0),
|
||
curPos: floCrypto.randInt(
|
||
0,
|
||
floGlobals.apiURL[floGlobals.blockchain].length
|
||
),
|
||
fetch_retry: function (apicall, rm_flosight) {
|
||
return new Promise((resolve, reject) => {
|
||
let i = this.serverList.indexOf(rm_flosight);
|
||
if (i != -1) this.serverList.splice(i, 1);
|
||
this.curPos = floCrypto.randInt(0, this.serverList.length);
|
||
this.fetch_api(apicall)
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
fetch_api: function (apicall) {
|
||
return new Promise((resolve, reject) => {
|
||
if (this.serverList.length === 0)
|
||
reject("No floSight server working");
|
||
else {
|
||
let flosight = this.serverList[this.curPos];
|
||
fetch(flosight + apicall)
|
||
.then((response) => {
|
||
if (response.ok)
|
||
response.json().then((data) => resolve(data));
|
||
else {
|
||
this.fetch_retry(apicall, flosight)
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
}
|
||
})
|
||
.catch((error) => {
|
||
this.fetch_retry(apicall, flosight)
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
});
|
||
}
|
||
});
|
||
},
|
||
},
|
||
|
||
//Promised function to get data from API
|
||
promisedAPI: function (apicall) {
|
||
return new Promise((resolve, reject) => {
|
||
console.log(apicall);
|
||
this.util
|
||
.fetch_api(apicall)
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
//Get balance for the given Address
|
||
getBalance: function (addr) {
|
||
return new Promise((resolve, reject) => {
|
||
this.promisedAPI(`api/addr/${addr}/balance`)
|
||
.then((balance) => resolve(parseFloat(balance)))
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
//Write Data into blockchain
|
||
writeData: function (
|
||
senderAddr,
|
||
data,
|
||
privKey,
|
||
receiverAddr = floGlobals.adminID
|
||
) {
|
||
return new Promise((resolve, reject) => {
|
||
if (typeof data != "string") data = JSON.stringify(data);
|
||
this.sendTx(
|
||
senderAddr,
|
||
receiverAddr,
|
||
floGlobals.sendAmt,
|
||
privKey,
|
||
data
|
||
)
|
||
.then((txid) => resolve(txid))
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
//Send Tx to blockchain
|
||
sendTx: function (
|
||
senderAddr,
|
||
receiverAddr,
|
||
sendAmt,
|
||
privKey,
|
||
floData = ""
|
||
) {
|
||
return new Promise((resolve, reject) => {
|
||
if (!floCrypto.validateAddr(senderAddr))
|
||
reject(`Invalid address : ${senderAddr}`);
|
||
else if (!floCrypto.validateAddr(receiverAddr))
|
||
reject(`Invalid address : ${receiverAddr}`);
|
||
if (
|
||
privKey.length < 1 ||
|
||
!floCrypto.verifyPrivKey(privKey, senderAddr)
|
||
)
|
||
reject("Invalid Private key!");
|
||
else if (typeof sendAmt !== "number" || sendAmt <= 0)
|
||
reject(`Invalid sendAmt : ${sendAmt}`);
|
||
else {
|
||
var trx = bitjs.transaction();
|
||
var utxoAmt = 0.0;
|
||
var fee = floGlobals.fee;
|
||
this.promisedAPI(`api/addr/${senderAddr}/utxo`)
|
||
.then((utxos) => {
|
||
for (
|
||
var i = utxos.length - 1;
|
||
i >= 0 && utxoAmt < sendAmt + fee;
|
||
i--
|
||
) {
|
||
if (utxos[i].confirmations) {
|
||
trx.addinput(
|
||
utxos[i].txid,
|
||
utxos[i].vout,
|
||
utxos[i].scriptPubKey
|
||
);
|
||
utxoAmt += utxos[i].amount;
|
||
} else break;
|
||
}
|
||
if (utxoAmt < sendAmt + fee) reject("Insufficient balance!");
|
||
else {
|
||
trx.addoutput(receiverAddr, sendAmt);
|
||
var change = utxoAmt - sendAmt - fee;
|
||
if (change > 0) trx.addoutput(senderAddr, change);
|
||
trx.addflodata(floData);
|
||
var signedTxHash = trx.sign(privKey, 1);
|
||
this.broadcastTx(signedTxHash)
|
||
.then((txid) => resolve(txid))
|
||
.catch((error) => reject(error));
|
||
}
|
||
})
|
||
.catch((error) => reject(error));
|
||
}
|
||
});
|
||
},
|
||
|
||
//merge all UTXOs of a given floID into a single UTXO
|
||
mergeUTXOs: function (floID, privKey, floData = "") {
|
||
return new Promise((resolve, reject) => {
|
||
if (!floCrypto.validateAddr(floID)) return reject(`Invalid floID`);
|
||
if (!floCrypto.verifyPrivKey(privKey, floID))
|
||
return reject("Invalid Private Key");
|
||
|
||
var trx = bitjs.transaction();
|
||
var utxoAmt = 0.0;
|
||
var fee = floGlobals.fee;
|
||
this.promisedAPI(`api/addr/${floID}/utxo`)
|
||
.then((utxos) => {
|
||
for (var i = utxos.length - 1; i >= 0; i--) {
|
||
if (utxos[i].confirmations) {
|
||
trx.addinput(
|
||
utxos[i].txid,
|
||
utxos[i].vout,
|
||
utxos[i].scriptPubKey
|
||
);
|
||
utxoAmt += utxos[i].amount;
|
||
}
|
||
}
|
||
trx.addoutput(floID, utxoAmt - fee);
|
||
trx.addflodata(floData);
|
||
var signedTxHash = trx.sign(privKey, 1);
|
||
this.broadcastTx(signedTxHash)
|
||
.then((txid) => resolve(txid))
|
||
.catch((error) => reject(error));
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
/**Write data into blockchain from (and/or) to multiple floID
|
||
* @param {Array} senderPrivKeys List of sender private-keys
|
||
* @param {string} data FLO data of the txn
|
||
* @param {Array} receivers List of receivers
|
||
* @param {boolean} preserveRatio (optional) preserve ratio or equal contribution
|
||
* @return {Promise}
|
||
*/
|
||
writeDataMultiple: function (
|
||
senderPrivKeys,
|
||
data,
|
||
receivers = [floGlobals.adminID],
|
||
preserveRatio = true
|
||
) {
|
||
return new Promise((resolve, reject) => {
|
||
if (!Array.isArray(senderPrivKeys))
|
||
return reject(
|
||
"Invalid senderPrivKeys: SenderPrivKeys must be Array"
|
||
);
|
||
if (!preserveRatio) {
|
||
let tmp = {};
|
||
let amount =
|
||
(floGlobals.sendAmt * receivers.length) / senderPrivKeys.length;
|
||
senderPrivKeys.forEach((key) => (tmp[key] = amount));
|
||
senderPrivKeys = tmp;
|
||
}
|
||
if (!Array.isArray(receivers))
|
||
return reject("Invalid receivers: Receivers must be Array");
|
||
else {
|
||
let tmp = {};
|
||
let amount = floGlobals.sendAmt;
|
||
receivers.forEach((floID) => (tmp[floID] = amount));
|
||
receivers = tmp;
|
||
}
|
||
if (typeof data != "string") data = JSON.stringify(data);
|
||
this.sendTxMultiple(senderPrivKeys, receivers, data)
|
||
.then((txid) => resolve(txid))
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
/**Send Tx from (and/or) to multiple floID
|
||
* @param {Array or Object} senderPrivKeys List of sender private-key (optional: with coins to be sent)
|
||
* @param {Object} receivers List of receivers with respective amount to be sent
|
||
* @param {string} floData FLO data of the txn
|
||
* @return {Promise}
|
||
*/
|
||
sendTxMultiple: function (senderPrivKeys, receivers, floData = "") {
|
||
return new Promise((resolve, reject) => {
|
||
let senders = {},
|
||
preserveRatio;
|
||
//check for argument validations
|
||
try {
|
||
let invalids = {
|
||
InvalidSenderPrivKeys: [],
|
||
InvalidSenderAmountFor: [],
|
||
InvalidReceiverIDs: [],
|
||
InvalidReceiveAmountFor: [],
|
||
};
|
||
let inputVal = 0,
|
||
outputVal = 0;
|
||
//Validate sender privatekeys (and send amount if passed)
|
||
//conversion when only privateKeys are passed (preserveRatio mode)
|
||
if (Array.isArray(senderPrivKeys)) {
|
||
senderPrivKeys.forEach((key) => {
|
||
try {
|
||
if (!key) invalids.InvalidSenderPrivKeys.push(key);
|
||
else {
|
||
let floID = floCrypto.getFloIDfromPubkeyHex(
|
||
floCrypto.getPubKeyHex(key)
|
||
);
|
||
senders[floID] = {
|
||
wif: key,
|
||
};
|
||
}
|
||
} catch (error) {
|
||
invalids.InvalidSenderPrivKeys.push(key);
|
||
}
|
||
});
|
||
preserveRatio = true;
|
||
}
|
||
//conversion when privatekeys are passed with send amount
|
||
else {
|
||
for (let key in senderPrivKeys) {
|
||
try {
|
||
if (!key) invalids.InvalidSenderPrivKeys.push(key);
|
||
else {
|
||
if (
|
||
typeof senderPrivKeys[key] !== "number" ||
|
||
senderPrivKeys[key] <= 0
|
||
)
|
||
invalids.InvalidSenderAmountFor.push(key);
|
||
else inputVal += senderPrivKeys[key];
|
||
let floID = floCrypto.getFloIDfromPubkeyHex(
|
||
floCrypto.getPubKeyHex(key)
|
||
);
|
||
senders[floID] = {
|
||
wif: key,
|
||
coins: senderPrivKeys[key],
|
||
};
|
||
}
|
||
} catch (error) {
|
||
invalids.InvalidSenderPrivKeys.push(key);
|
||
}
|
||
}
|
||
preserveRatio = false;
|
||
}
|
||
//Validate the receiver IDs and receive amount
|
||
for (let floID in receivers) {
|
||
if (!floCrypto.validateAddr(floID))
|
||
invalids.InvalidReceiverIDs.push(floID);
|
||
if (typeof receivers[floID] !== "number" || receivers[floID] <= 0)
|
||
invalids.InvalidReceiveAmountFor.push(floID);
|
||
else outputVal += receivers[floID];
|
||
}
|
||
//Reject if any invalids are found
|
||
for (let i in invalids) if (!invalids[i].length) delete invalids[i];
|
||
if (Object.keys(invalids).length) return reject(invalids);
|
||
//Reject if given inputVal and outputVal are not equal
|
||
if (!preserveRatio && inputVal != outputVal)
|
||
return reject(
|
||
`Input Amount (${inputVal}) not equal to Output Amount (${outputVal})`
|
||
);
|
||
} catch (error) {
|
||
return reject(error);
|
||
}
|
||
//Get balance of senders
|
||
let promises = [];
|
||
for (let floID in senders) promises.push(this.getBalance(floID));
|
||
Promise.all(promises)
|
||
.then((results) => {
|
||
let totalBalance = 0,
|
||
totalFee = floGlobals.fee,
|
||
balance = {};
|
||
//Divide fee among sender if not for preserveRatio
|
||
if (!preserveRatio)
|
||
var dividedFee = totalFee / Object.keys(senders).length;
|
||
//Check if balance of each sender is sufficient enough
|
||
let insufficient = [];
|
||
for (let floID in senders) {
|
||
balance[floID] = parseFloat(results.shift());
|
||
if (
|
||
isNaN(balance[floID]) ||
|
||
(preserveRatio && balance[floID] <= totalFee) ||
|
||
(!preserveRatio &&
|
||
balance[floID] < senders[floID].coins + dividedFee)
|
||
)
|
||
insufficient.push(floID);
|
||
totalBalance += balance[floID];
|
||
}
|
||
if (insufficient.length)
|
||
return reject({
|
||
InsufficientBalance: insufficient,
|
||
});
|
||
//Calculate totalSentAmount and check if totalBalance is sufficient
|
||
let totalSendAmt = totalFee;
|
||
for (floID in receivers) totalSendAmt += receivers[floID];
|
||
if (totalBalance < totalSendAmt)
|
||
return reject("Insufficient total Balance");
|
||
//Get the UTXOs of the senders
|
||
let promises = [];
|
||
for (floID in senders)
|
||
promises.push(this.promisedAPI(`api/addr/${floID}/utxo`));
|
||
Promise.all(promises)
|
||
.then((results) => {
|
||
let wifSeq = [];
|
||
var trx = bitjs.transaction();
|
||
for (floID in senders) {
|
||
let utxos = results.shift();
|
||
let sendAmt;
|
||
if (preserveRatio) {
|
||
let ratio = balance[floID] / totalBalance;
|
||
sendAmt = totalSendAmt * ratio;
|
||
} else sendAmt = senders[floID].coins + dividedFee;
|
||
let wif = senders[floID].wif;
|
||
let utxoAmt = 0.0;
|
||
for (
|
||
let i = utxos.length - 1;
|
||
i >= 0 && utxoAmt < sendAmt;
|
||
i--
|
||
) {
|
||
if (utxos[i].confirmations) {
|
||
trx.addinput(
|
||
utxos[i].txid,
|
||
utxos[i].vout,
|
||
utxos[i].scriptPubKey
|
||
);
|
||
wifSeq.push(wif);
|
||
utxoAmt += utxos[i].amount;
|
||
}
|
||
}
|
||
if (utxoAmt < sendAmt)
|
||
return reject("Insufficient balance:" + floID);
|
||
let change = utxoAmt - sendAmt;
|
||
if (change > 0) trx.addoutput(floID, change);
|
||
}
|
||
for (floID in receivers)
|
||
trx.addoutput(floID, receivers[floID]);
|
||
trx.addflodata(floData);
|
||
for (let i = 0; i < wifSeq.length; i++)
|
||
trx.signinput(i, wifSeq[i], 1);
|
||
var signedTxHash = trx.serialize();
|
||
this.broadcastTx(signedTxHash)
|
||
.then((txid) => resolve(txid))
|
||
.catch((error) => reject(error));
|
||
})
|
||
.catch((error) => reject(error));
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
//Broadcast signed Tx in blockchain using API
|
||
broadcastTx: function (signedTxHash) {
|
||
return new Promise((resolve, reject) => {
|
||
var request = new XMLHttpRequest();
|
||
var url = this.util.serverList[this.util.curPos] + "api/tx/send";
|
||
console.log(url);
|
||
if (signedTxHash.length < 1) reject("Empty Signature");
|
||
else {
|
||
var params = `{"rawtx":"${signedTxHash}"}`;
|
||
request.open("POST", url, true);
|
||
//Send the proper header information along with the request
|
||
request.setRequestHeader("Content-type", "application/json");
|
||
request.onload = function () {
|
||
if (request.readyState == 4 && request.status == 200) {
|
||
console.log(request.response);
|
||
resolve(JSON.parse(request.response).txid.result);
|
||
} else reject(request.responseText);
|
||
};
|
||
request.send(params);
|
||
}
|
||
});
|
||
},
|
||
|
||
//Read Txs of Address between from and to
|
||
readTxs: function (addr, from, to) {
|
||
return new Promise((resolve, reject) => {
|
||
this.promisedAPI(`api/addrs/${addr}/txs?from=${from}&to=${to}`)
|
||
.then((response) => resolve(response))
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
//Read All Txs of Address (newest first)
|
||
readAllTxs: function (addr) {
|
||
return new Promise((resolve, reject) => {
|
||
this.promisedAPI(`api/addrs/${addr}/txs?from=0&to=1`)
|
||
.then((response) => {
|
||
this.promisedAPI(
|
||
`api/addrs/${addr}/txs?from=0&to=${response.totalItems}0`
|
||
)
|
||
.then((response) => resolve(response.items))
|
||
.catch((error) => reject(error));
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
/*Read flo Data from txs of given Address
|
||
options can be used to filter data
|
||
limit : maximum number of filtered data (default = 1000, negative = no limit)
|
||
ignoreOld : ignore old txs (default = 0)
|
||
sentOnly : filters only sent data
|
||
pattern : filters data that starts with a pattern
|
||
contains : filters data that contains a string
|
||
filter : custom filter funtion for floData (eg . filter: d => {return d[0] == '$'})
|
||
*/
|
||
readData: function (addr, options = {}) {
|
||
options.limit = options.limit | 1000;
|
||
options.ignoreOld = options.ignoreOld | 0;
|
||
return new Promise((resolve, reject) => {
|
||
this.promisedAPI(`api/addrs/${addr}/txs?from=0&to=1`)
|
||
.then((response) => {
|
||
var newItems = response.totalItems - options.ignoreOld;
|
||
this.promisedAPI(
|
||
`api/addrs/${addr}/txs?from=0&to=${newItems * 2}`
|
||
)
|
||
.then((response) => {
|
||
if (options.limit <= 0) options.limit = response.items.length;
|
||
var filteredData = [];
|
||
for (
|
||
i = 0;
|
||
i < response.totalItems - options.ignoreOld &&
|
||
filteredData.length < options.limit;
|
||
i++
|
||
) {
|
||
if (
|
||
options.sentOnly &&
|
||
response.items[i].vin[0].addr !== addr
|
||
)
|
||
continue;
|
||
if (
|
||
options.pattern &&
|
||
!response.items[i].floData.startsWith(
|
||
options.pattern,
|
||
0
|
||
) &&
|
||
!response.items[i].floData.startsWith(options.pattern, 2)
|
||
)
|
||
continue;
|
||
if (
|
||
options.contains &&
|
||
!response.items[i].floData.includes(options.contains)
|
||
)
|
||
continue;
|
||
if (
|
||
options.filter &&
|
||
!options.filter(response.items[i].floData)
|
||
)
|
||
continue;
|
||
filteredData.push(response.items[i].floData);
|
||
}
|
||
resolve({
|
||
totalTxs: response.totalItems,
|
||
data: filteredData,
|
||
});
|
||
})
|
||
.catch((error) => {
|
||
reject(error);
|
||
});
|
||
})
|
||
.catch((error) => {
|
||
reject(error);
|
||
});
|
||
});
|
||
},
|
||
};
|
||
</script>
|
||
|
||
<script id="floSupernode">
|
||
/* flo Supernode Operators to send/receive data from supernodes using websocket */
|
||
const floSupernode = {
|
||
//kBucket object
|
||
kBucket: {
|
||
supernodeKBucket: null,
|
||
decodeBase58Address: function (address) {
|
||
let k = bitjs.Base58.decode(address);
|
||
k.shift();
|
||
k.splice(-4, 4);
|
||
return Crypto.util.bytesToHex(k);
|
||
},
|
||
floIdToKbucketId: function (address) {
|
||
const decodedId = this.decodeBase58Address(address);
|
||
const nodeIdBigInt = new BigInteger(decodedId, 16);
|
||
const nodeIdBytes = nodeIdBigInt.toByteArrayUnsigned();
|
||
const nodeIdNewInt8Array = new Uint8Array(nodeIdBytes);
|
||
return nodeIdNewInt8Array;
|
||
},
|
||
launch: function (
|
||
superNodeList = Object.keys(floGlobals.supernodes),
|
||
master_floID = floGlobals.adminID
|
||
) {
|
||
return new Promise((resolve, reject) => {
|
||
try {
|
||
const SuKBucketId = this.floIdToKbucketId(master_floID);
|
||
const SukbOptions = {
|
||
localNodeId: SuKBucketId,
|
||
};
|
||
this.supernodeKBucket = new BuildKBucket(SukbOptions);
|
||
for (var i = 0; i < superNodeList.length; i++) {
|
||
this.addNewNode(superNodeList[i]);
|
||
}
|
||
resolve("SuperNode KBucket formed");
|
||
} catch (error) {
|
||
reject(error);
|
||
}
|
||
});
|
||
},
|
||
addContact: function (id, floID, KB = this.supernodeKBucket) {
|
||
const contact = {
|
||
id: id,
|
||
floID: floID,
|
||
};
|
||
KB.add(contact);
|
||
},
|
||
addNewNode: function (address, KB = this.supernodeKBucket) {
|
||
let decodedId = address;
|
||
try {
|
||
decodedId = this.floIdToKbucketId(address);
|
||
} catch (e) {
|
||
decodedId = address;
|
||
}
|
||
this.addContact(decodedId, address, KB);
|
||
},
|
||
isNodePresent: function (flo_id, KB = this.supernodeKBucket) {
|
||
let kArray = KB.toArray();
|
||
let kArrayFloIds = kArray.map((k) => k.floID);
|
||
if (kArrayFloIds.includes(flo_id)) return true;
|
||
else return false;
|
||
},
|
||
|
||
getInnerNodes: function (
|
||
flo_addr1,
|
||
flo_addr2,
|
||
KB = this.supernodeKBucket
|
||
) {
|
||
return new Promise((resolve, reject) => {
|
||
let kArrayFloIds = KB.toArray().map((k) => k.floID);
|
||
var innerNodes = [];
|
||
if (
|
||
kArrayFloIds.includes(flo_addr1) &&
|
||
kArrayFloIds.includes(flo_addr2)
|
||
) {
|
||
for (
|
||
var i = kArrayFloIds.indexOf(flo_addr1);
|
||
i != flo_addr2;
|
||
i++
|
||
) {
|
||
if (i >= kArrayFloIds.length) i = -1;
|
||
else innerNodes.push(kArrayFloIds[i]);
|
||
}
|
||
resolve(innerNodes);
|
||
} else reject("Given nodes are not in KBucket");
|
||
});
|
||
},
|
||
|
||
getOuterNodes: function (
|
||
flo_addr1,
|
||
flo_addr2,
|
||
KB = this.supernodeKBucket
|
||
) {
|
||
return new Promise((resolve, reject) => {
|
||
let kArrayFloIds = KB.toArray().map((k) => k.floID);
|
||
var outterNodes = [];
|
||
if (
|
||
kArrayFloIds.includes(flo_addr1) &&
|
||
kArrayFloIds.includes(flo_addr2)
|
||
) {
|
||
for (
|
||
var i = kArrayFloIds.indexOf(flo_addr2);
|
||
i != flo_addr1;
|
||
i++
|
||
) {
|
||
if (i >= kArrayFloIds.length) i = -1;
|
||
else outterNodes.push(kArrayFloIds[i]);
|
||
}
|
||
resolve(outterNodes);
|
||
} else reject("Given nodes are not in KBucket");
|
||
});
|
||
},
|
||
|
||
getPrevSupernode: function (
|
||
flo_addr,
|
||
n = 1,
|
||
KB = this.supernodeKBucket
|
||
) {
|
||
return new Promise((resolve, reject) => {
|
||
try {
|
||
let kArrayFloIds = KB.toArray().map((k) => k.floID);
|
||
let pos = kArrayFloIds.indexOf(flo_addr);
|
||
var prevSupernode = [];
|
||
for (var i = 1; i <= n; i++) {
|
||
if (pos - i < 0) var prev = pos - i + kArrayFloIds.length;
|
||
else var prev = pos - i;
|
||
prevSupernode.push(kArrayFloIds[prev]);
|
||
}
|
||
resolve(prevSupernode);
|
||
} catch (error) {
|
||
reject(error);
|
||
}
|
||
});
|
||
},
|
||
|
||
getNextSupernode: function (
|
||
flo_addr,
|
||
n = 1,
|
||
KB = this.supernodeKBucket
|
||
) {
|
||
return new Promise((resolve, reject) => {
|
||
try {
|
||
let kArrayFloIds = KB.toArray().map((k) => k.floID);
|
||
let pos = kArrayFloIds.indexOf(flo_addr);
|
||
var nextSupernode = [];
|
||
for (var i = 1; i <= n; i++) {
|
||
if (pos + i >= kArrayFloIds.length)
|
||
var next = pos + i - kArrayFloIds.length;
|
||
else var next = pos + i;
|
||
nextSupernode.push(kArrayFloIds[next]);
|
||
}
|
||
resolve(nextSupernode);
|
||
} catch (error) {
|
||
reject(error);
|
||
}
|
||
});
|
||
},
|
||
|
||
determineClosestSupernode: function (
|
||
flo_addr,
|
||
n = 1,
|
||
KB = this.supernodeKBucket
|
||
) {
|
||
return new Promise((resolve, reject) => {
|
||
try {
|
||
let isFloIdUint8 = flo_addr instanceof Uint8Array;
|
||
if (!isFloIdUint8) flo_addr = this.floIdToKbucketId(flo_addr);
|
||
var closestSupernode = KB.closest(flo_addr, n);
|
||
closestSupernode = closestSupernode.map((k) => k.floID);
|
||
resolve(closestSupernode);
|
||
} catch (error) {
|
||
reject(error);
|
||
}
|
||
});
|
||
},
|
||
},
|
||
|
||
sendDataToSN(data, snfloID) {
|
||
return new Promise((resolve, reject) => {
|
||
var websocket = new WebSocket(
|
||
"wss://" + floGlobals.supernodes[snfloID].uri + "/ws"
|
||
);
|
||
websocket.onmessage = (evt) => {
|
||
if (evt.data == "$+") {
|
||
websocket.send(data);
|
||
resolve(`Data sent to supernode : ${snfloID}`);
|
||
} else if (evt.data == "$-") {
|
||
this.kBucket
|
||
.getNextSupernode(snfloID)
|
||
.then((nextNode) => {
|
||
this.sendDataToSN(data, nextNode[0])
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
})
|
||
.catch((error) => reject(error));
|
||
} else {
|
||
console.log(evt.data);
|
||
reject(evt.data);
|
||
}
|
||
websocket.close();
|
||
};
|
||
websocket.onerror = (evt) => {
|
||
this.kBucket
|
||
.getNextSupernode(snfloID)
|
||
.then((nextNode) => {
|
||
this.sendDataToSN(data, nextNode[0])
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
})
|
||
.catch((error) => reject(error));
|
||
};
|
||
});
|
||
},
|
||
|
||
//Sends data to the supernode
|
||
sendData: function (data, floID) {
|
||
return new Promise((resolve, reject) => {
|
||
this.kBucket
|
||
.determineClosestSupernode(floID)
|
||
.then((closestNode) => {
|
||
this.sendDataToSN(data, closestNode[0])
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
})
|
||
.catch((error) => {
|
||
reject(error);
|
||
});
|
||
});
|
||
},
|
||
|
||
requestDataFromSN(request, snfloID) {
|
||
return new Promise((resolve, reject) => {
|
||
var websocket = new WebSocket(
|
||
"wss://" + floGlobals.supernodes[snfloID].uri + "/ws"
|
||
);
|
||
websocket.onmessage = (evt) => {
|
||
if (evt.data == "$+") {
|
||
websocket.send(`?${request}`);
|
||
} else if (evt.data == "$-") {
|
||
this.kBucket
|
||
.getNextSupernode(snfloID)
|
||
.then((nextNode) => {
|
||
this.requestDataFromSN(request, nextNode[0])
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
})
|
||
.catch((error) => reject(error));
|
||
websocket.close();
|
||
} else {
|
||
resolve(evt.data);
|
||
websocket.close();
|
||
}
|
||
};
|
||
websocket.onerror = (evt) => {
|
||
this.kBucket
|
||
.getNextSupernode(snfloID)
|
||
.then((nextNode) => {
|
||
this.requestDataFromSN(request, nextNode[0])
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
})
|
||
.catch((error) => reject(error));
|
||
};
|
||
});
|
||
},
|
||
|
||
//Request data from supernode
|
||
requestData: function (request, floID) {
|
||
return new Promise((resolve, reject) => {
|
||
this.kBucket
|
||
.determineClosestSupernode(floID)
|
||
.then((closestNode) => {
|
||
this.requestDataFromSN(request, closestNode[0])
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
})
|
||
.catch((error) => {
|
||
reject(error);
|
||
});
|
||
});
|
||
},
|
||
};
|
||
|
||
/**Kademlia DHT K-bucket implementation as a binary tree.
|
||
* Implementation of a Kademlia DHT k-bucket used for storing
|
||
* contact (peer node) information.
|
||
*
|
||
* @extends EventEmitter
|
||
*/
|
||
function BuildKBucket(options = {}) {
|
||
/**
|
||
* `options`:
|
||
* `distance`: Function
|
||
* `function (firstId, secondId) { return distance }` An optional
|
||
* `distance` function that gets two `id` Uint8Arrays
|
||
* and return distance (as number) between them.
|
||
* `arbiter`: Function (Default: vectorClock arbiter)
|
||
* `function (incumbent, candidate) { return contact; }` An optional
|
||
* `arbiter` function that givent two `contact` objects with the same `id`
|
||
* returns the desired object to be used for updating the k-bucket. For
|
||
* more details, see [arbiter function](#arbiter-function).
|
||
* `localNodeId`: Uint8Array An optional Uint8Array representing the local node id.
|
||
* If not provided, a local node id will be created via `randomBytes(20)`.
|
||
* `metadata`: Object (Default: {}) Optional satellite data to include
|
||
* with the k-bucket. `metadata` property is guaranteed not be altered by,
|
||
* it is provided as an explicit container for users of k-bucket to store
|
||
* implementation-specific data.
|
||
* `numberOfNodesPerKBucket`: Integer (Default: 20) The number of nodes
|
||
* that a k-bucket can contain before being full or split.
|
||
* `numberOfNodesToPing`: Integer (Default: 3) The number of nodes to
|
||
* ping when a bucket that should not be split becomes full. KBucket will
|
||
* emit a `ping` event that contains `numberOfNodesToPing` nodes that have
|
||
* not been contacted the longest.
|
||
*
|
||
* @param {Object=} options optional
|
||
*/
|
||
|
||
this.localNodeId =
|
||
options.localNodeId ||
|
||
window.crypto.getRandomValues(new Uint8Array(20));
|
||
this.numberOfNodesPerKBucket = options.numberOfNodesPerKBucket || 20;
|
||
this.numberOfNodesToPing = options.numberOfNodesToPing || 3;
|
||
this.distance = options.distance || this.distance;
|
||
// use an arbiter from options or vectorClock arbiter by default
|
||
this.arbiter = options.arbiter || this.arbiter;
|
||
this.metadata = Object.assign({}, options.metadata);
|
||
|
||
this.createNode = function () {
|
||
return {
|
||
contacts: [],
|
||
dontSplit: false,
|
||
left: null,
|
||
right: null,
|
||
};
|
||
};
|
||
|
||
this.ensureInt8 = function (name, val) {
|
||
if (!(val instanceof Uint8Array)) {
|
||
throw new TypeError(name + " is not a Uint8Array");
|
||
}
|
||
};
|
||
|
||
/**
|
||
* @param {Uint8Array} array1
|
||
* @param {Uint8Array} array2
|
||
* @return {Boolean}
|
||
*/
|
||
this.arrayEquals = function (array1, array2) {
|
||
if (array1 === array2) {
|
||
return true;
|
||
}
|
||
if (array1.length !== array2.length) {
|
||
return false;
|
||
}
|
||
for (let i = 0, length = array1.length; i < length; ++i) {
|
||
if (array1[i] !== array2[i]) {
|
||
return false;
|
||
}
|
||
}
|
||
return true;
|
||
};
|
||
|
||
this.ensureInt8("option.localNodeId as parameter 1", this.localNodeId);
|
||
this.root = this.createNode();
|
||
|
||
/**
|
||
* Default arbiter function for contacts with the same id. Uses
|
||
* contact.vectorClock to select which contact to update the k-bucket with.
|
||
* Contact with larger vectorClock field will be selected. If vectorClock is
|
||
* the same, candidat will be selected.
|
||
*
|
||
* @param {Object} incumbent Contact currently stored in the k-bucket.
|
||
* @param {Object} candidate Contact being added to the k-bucket.
|
||
* @return {Object} Contact to updated the k-bucket with.
|
||
*/
|
||
this.arbiter = function (incumbent, candidate) {
|
||
return incumbent.vectorClock > candidate.vectorClock
|
||
? incumbent
|
||
: candidate;
|
||
};
|
||
|
||
/**
|
||
* Default distance function. Finds the XOR
|
||
* distance between firstId and secondId.
|
||
*
|
||
* @param {Uint8Array} firstId Uint8Array containing first id.
|
||
* @param {Uint8Array} secondId Uint8Array containing second id.
|
||
* @return {Number} Integer The XOR distance between firstId
|
||
* and secondId.
|
||
*/
|
||
this.distance = function (firstId, secondId) {
|
||
let distance = 0;
|
||
let i = 0;
|
||
const min = Math.min(firstId.length, secondId.length);
|
||
const max = Math.max(firstId.length, secondId.length);
|
||
for (; i < min; ++i) {
|
||
distance = distance * 256 + (firstId[i] ^ secondId[i]);
|
||
}
|
||
for (; i < max; ++i) distance = distance * 256 + 255;
|
||
return distance;
|
||
};
|
||
|
||
/**
|
||
* Adds a contact to the k-bucket.
|
||
*
|
||
* @param {Object} contact the contact object to add
|
||
*/
|
||
this.add = function (contact) {
|
||
this.ensureInt8("contact.id", (contact || {}).id);
|
||
|
||
let bitIndex = 0;
|
||
let node = this.root;
|
||
|
||
while (node.contacts === null) {
|
||
// this is not a leaf node but an inner node with 'low' and 'high'
|
||
// branches; we will check the appropriate bit of the identifier and
|
||
// delegate to the appropriate node for further processing
|
||
node = this._determineNode(node, contact.id, bitIndex++);
|
||
}
|
||
|
||
// check if the contact already exists
|
||
const index = this._indexOf(node, contact.id);
|
||
if (index >= 0) {
|
||
this._update(node, index, contact);
|
||
return this;
|
||
}
|
||
|
||
if (node.contacts.length < this.numberOfNodesPerKBucket) {
|
||
node.contacts.push(contact);
|
||
return this;
|
||
}
|
||
|
||
// the bucket is full
|
||
if (node.dontSplit) {
|
||
// we are not allowed to split the bucket
|
||
// we need to ping the first this.numberOfNodesToPing
|
||
// in order to determine if they are alive
|
||
// only if one of the pinged nodes does not respond, can the new contact
|
||
// be added (this prevents DoS flodding with new invalid contacts)
|
||
return this;
|
||
}
|
||
|
||
this._split(node, bitIndex);
|
||
return this.add(contact);
|
||
};
|
||
|
||
/**
|
||
* Get the n closest contacts to the provided node id. "Closest" here means:
|
||
* closest according to the XOR metric of the contact node id.
|
||
*
|
||
* @param {Uint8Array} id Contact node id
|
||
* @param {Number=} n Integer (Default: Infinity) The maximum number of
|
||
* closest contacts to return
|
||
* @return {Array} Array Maximum of n closest contacts to the node id
|
||
*/
|
||
this.closest = function (id, n = Infinity) {
|
||
this.ensureInt8("id", id);
|
||
|
||
if ((!Number.isInteger(n) && n !== Infinity) || n <= 0) {
|
||
throw new TypeError("n is not positive number");
|
||
}
|
||
|
||
let contacts = [];
|
||
|
||
for (
|
||
let nodes = [this.root], bitIndex = 0;
|
||
nodes.length > 0 && contacts.length < n;
|
||
|
||
) {
|
||
const node = nodes.pop();
|
||
if (node.contacts === null) {
|
||
const detNode = this._determineNode(node, id, bitIndex++);
|
||
nodes.push(node.left === detNode ? node.right : node.left);
|
||
nodes.push(detNode);
|
||
} else {
|
||
contacts = contacts.concat(node.contacts);
|
||
}
|
||
}
|
||
|
||
return contacts
|
||
.map((a) => [this.distance(a.id, id), a])
|
||
.sort((a, b) => a[0] - b[0])
|
||
.slice(0, n)
|
||
.map((a) => a[1]);
|
||
};
|
||
|
||
/**
|
||
* Counts the total number of contacts in the tree.
|
||
*
|
||
* @return {Number} The number of contacts held in the tree
|
||
*/
|
||
this.count = function () {
|
||
// return this.toArray().length
|
||
let count = 0;
|
||
for (const nodes = [this.root]; nodes.length > 0; ) {
|
||
const node = nodes.pop();
|
||
if (node.contacts === null) nodes.push(node.right, node.left);
|
||
else count += node.contacts.length;
|
||
}
|
||
return count;
|
||
};
|
||
|
||
/**
|
||
* Determines whether the id at the bitIndex is 0 or 1.
|
||
* Return left leaf if `id` at `bitIndex` is 0, right leaf otherwise
|
||
*
|
||
* @param {Object} node internal object that has 2 leafs: left and right
|
||
* @param {Uint8Array} id Id to compare localNodeId with.
|
||
* @param {Number} bitIndex Integer (Default: 0) The bit index to which bit
|
||
* to check in the id Uint8Array.
|
||
* @return {Object} left leaf if id at bitIndex is 0, right leaf otherwise.
|
||
*/
|
||
this._determineNode = function (node, id, bitIndex) {
|
||
// *NOTE* remember that id is a Uint8Array and has granularity of
|
||
// bytes (8 bits), whereas the bitIndex is the bit index (not byte)
|
||
|
||
// id's that are too short are put in low bucket (1 byte = 8 bits)
|
||
// (bitIndex >> 3) finds how many bytes the bitIndex describes
|
||
// bitIndex % 8 checks if we have extra bits beyond byte multiples
|
||
// if number of bytes is <= no. of bytes described by bitIndex and there
|
||
// are extra bits to consider, this means id has less bits than what
|
||
// bitIndex describes, id therefore is too short, and will be put in low
|
||
// bucket
|
||
const bytesDescribedByBitIndex = bitIndex >> 3;
|
||
const bitIndexWithinByte = bitIndex % 8;
|
||
if (id.length <= bytesDescribedByBitIndex && bitIndexWithinByte !== 0) {
|
||
return node.left;
|
||
}
|
||
|
||
const byteUnderConsideration = id[bytesDescribedByBitIndex];
|
||
|
||
// byteUnderConsideration is an integer from 0 to 255 represented by 8 bits
|
||
// where 255 is 11111111 and 0 is 00000000
|
||
// in order to find out whether the bit at bitIndexWithinByte is set
|
||
// we construct (1 << (7 - bitIndexWithinByte)) which will consist
|
||
// of all bits being 0, with only one bit set to 1
|
||
// for example, if bitIndexWithinByte is 3, we will construct 00010000 by
|
||
// (1 << (7 - 3)) -> (1 << 4) -> 16
|
||
if (byteUnderConsideration & (1 << (7 - bitIndexWithinByte))) {
|
||
return node.right;
|
||
}
|
||
|
||
return node.left;
|
||
};
|
||
|
||
/**
|
||
* Get a contact by its exact ID.
|
||
* If this is a leaf, loop through the bucket contents and return the correct
|
||
* contact if we have it or null if not. If this is an inner node, determine
|
||
* which branch of the tree to traverse and repeat.
|
||
*
|
||
* @param {Uint8Array} id The ID of the contact to fetch.
|
||
* @return {Object|Null} The contact if available, otherwise null
|
||
*/
|
||
this.get = function (id) {
|
||
this.ensureInt8("id", id);
|
||
|
||
let bitIndex = 0;
|
||
|
||
let node = this.root;
|
||
while (node.contacts === null) {
|
||
node = this._determineNode(node, id, bitIndex++);
|
||
}
|
||
|
||
// index of uses contact id for matching
|
||
const index = this._indexOf(node, id);
|
||
return index >= 0 ? node.contacts[index] : null;
|
||
};
|
||
|
||
/**
|
||
* Returns the index of the contact with provided
|
||
* id if it exists, returns -1 otherwise.
|
||
*
|
||
* @param {Object} node internal object that has 2 leafs: left and right
|
||
* @param {Uint8Array} id Contact node id.
|
||
* @return {Number} Integer Index of contact with provided id if it
|
||
* exists, -1 otherwise.
|
||
*/
|
||
this._indexOf = function (node, id) {
|
||
for (let i = 0; i < node.contacts.length; ++i) {
|
||
if (this.arrayEquals(node.contacts[i].id, id)) return i;
|
||
}
|
||
|
||
return -1;
|
||
};
|
||
|
||
/**
|
||
* Removes contact with the provided id.
|
||
*
|
||
* @param {Uint8Array} id The ID of the contact to remove.
|
||
* @return {Object} The k-bucket itself.
|
||
*/
|
||
this.remove = function (id) {
|
||
this.ensureInt8("the id as parameter 1", id);
|
||
|
||
let bitIndex = 0;
|
||
let node = this.root;
|
||
|
||
while (node.contacts === null) {
|
||
node = this._determineNode(node, id, bitIndex++);
|
||
}
|
||
|
||
const index = this._indexOf(node, id);
|
||
if (index >= 0) {
|
||
const contact = node.contacts.splice(index, 1)[0];
|
||
}
|
||
|
||
return this;
|
||
};
|
||
|
||
/**
|
||
* Splits the node, redistributes contacts to the new nodes, and marks the
|
||
* node that was split as an inner node of the binary tree of nodes by
|
||
* setting this.root.contacts = null
|
||
*
|
||
* @param {Object} node node for splitting
|
||
* @param {Number} bitIndex the bitIndex to which byte to check in the
|
||
* Uint8Array for navigating the binary tree
|
||
*/
|
||
this._split = function (node, bitIndex) {
|
||
node.left = this.createNode();
|
||
node.right = this.createNode();
|
||
|
||
// redistribute existing contacts amongst the two newly created nodes
|
||
for (const contact of node.contacts) {
|
||
this._determineNode(node, contact.id, bitIndex).contacts.push(
|
||
contact
|
||
);
|
||
}
|
||
|
||
node.contacts = null; // mark as inner tree node
|
||
|
||
// don't split the "far away" node
|
||
// we check where the local node would end up and mark the other one as
|
||
// "dontSplit" (i.e. "far away")
|
||
const detNode = this._determineNode(node, this.localNodeId, bitIndex);
|
||
const otherNode = node.left === detNode ? node.right : node.left;
|
||
otherNode.dontSplit = true;
|
||
};
|
||
|
||
/**
|
||
* Returns all the contacts contained in the tree as an array.
|
||
* If this is a leaf, return a copy of the bucket. `slice` is used so that we
|
||
* don't accidentally leak an internal reference out that might be
|
||
* accidentally misused. If this is not a leaf, return the union of the low
|
||
* and high branches (themselves also as arrays).
|
||
*
|
||
* @return {Array} All of the contacts in the tree, as an array
|
||
*/
|
||
this.toArray = function () {
|
||
let result = [];
|
||
for (const nodes = [this.root]; nodes.length > 0; ) {
|
||
const node = nodes.pop();
|
||
if (node.contacts === null) nodes.push(node.right, node.left);
|
||
else result = result.concat(node.contacts);
|
||
}
|
||
return result;
|
||
};
|
||
|
||
/**
|
||
* Updates the contact selected by the arbiter.
|
||
* If the selection is our old contact and the candidate is some new contact
|
||
* then the new contact is abandoned (not added).
|
||
* If the selection is our old contact and the candidate is our old contact
|
||
* then we are refreshing the contact and it is marked as most recently
|
||
* contacted (by being moved to the right/end of the bucket array).
|
||
* If the selection is our new contact, the old contact is removed and the new
|
||
* contact is marked as most recently contacted.
|
||
*
|
||
* @param {Object} node internal object that has 2 leafs: left and right
|
||
* @param {Number} index the index in the bucket where contact exists
|
||
* (index has already been computed in a previous
|
||
* calculation)
|
||
* @param {Object} contact The contact object to update.
|
||
*/
|
||
this._update = function (node, index, contact) {
|
||
// sanity check
|
||
if (!this.arrayEquals(node.contacts[index].id, contact.id)) {
|
||
throw new Error("wrong index for _update");
|
||
}
|
||
|
||
const incumbent = node.contacts[index];
|
||
const selection = this.arbiter(incumbent, contact);
|
||
// if the selection is our old contact and the candidate is some new
|
||
// contact, then there is nothing to do
|
||
if (selection === incumbent && incumbent !== contact) return;
|
||
|
||
node.contacts.splice(index, 1); // remove old contact
|
||
node.contacts.push(selection); // add more recent contact version
|
||
};
|
||
}
|
||
</script>
|
||
|
||
<script id="compactIDB">
|
||
/* Compact IndexedDB operations */
|
||
|
||
window.indexedDB =
|
||
window.indexedDB ||
|
||
window.mozIndexedDB ||
|
||
window.webkitIndexedDB ||
|
||
window.msIndexedDB;
|
||
window.IDBTransaction =
|
||
window.IDBTransaction ||
|
||
window.webkitIDBTransaction ||
|
||
window.msIDBTransaction;
|
||
window.IDBKeyRange =
|
||
window.IDBKeyRange || window.webkitIDBKeyRange || window.msIDBKeyRange;
|
||
|
||
if (!window.indexedDB)
|
||
window.alert(
|
||
"Your browser doesn't support a stable version of IndexedDB."
|
||
);
|
||
|
||
const compactIDB = {
|
||
setDefaultDB: function (dbName) {
|
||
this.defaultDB = dbName;
|
||
},
|
||
|
||
initDB: function (
|
||
dbName,
|
||
objectStores = {},
|
||
version = null,
|
||
removeStores = []
|
||
) {
|
||
return new Promise((resolve, reject) => {
|
||
this.defaultDB = this.defaultDB || dbName;
|
||
var idb = version
|
||
? indexedDB.open(dbName, version)
|
||
: indexedDB.open(dbName);
|
||
idb.onerror = (event) => {
|
||
reject("Error in opening IndexedDB!");
|
||
};
|
||
idb.onupgradeneeded = (event) => {
|
||
var db = event.target.result;
|
||
for (let obs in objectStores) {
|
||
var objectStore = db.createObjectStore(
|
||
obs,
|
||
objectStores[obs].options || {}
|
||
);
|
||
if (
|
||
objectStores[obs].indexes &&
|
||
typeof objectStores[obs].indexes === "object"
|
||
)
|
||
for (let i in objectStores[obs].indexes)
|
||
objectStore.createIndex(
|
||
i,
|
||
i,
|
||
objectStores[obs].indexes[i] || {}
|
||
);
|
||
}
|
||
if (version)
|
||
removeStores.forEach((obs) => db.deleteObjectStore(obs));
|
||
};
|
||
idb.onsuccess = (event) => {
|
||
var db = event.target.result;
|
||
if (
|
||
JSON.stringify(Object.values(db.objectStoreNames).sort()) ===
|
||
JSON.stringify(Object.keys(objectStores).sort())
|
||
)
|
||
resolve("Initiated IndexedDB");
|
||
else {
|
||
let removeObs = [];
|
||
Object.values(db.objectStoreNames).forEach((obs) => {
|
||
if (obs in objectStores) delete objectStores[obs];
|
||
else removeObs.push(obs);
|
||
});
|
||
this.initDB(dbName, objectStores, db.version + 1, removeObs)
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
}
|
||
db.close();
|
||
};
|
||
});
|
||
},
|
||
|
||
openDB: function (dbName = this.defaultDB) {
|
||
return new Promise((resolve, reject) => {
|
||
var idb = indexedDB.open(dbName);
|
||
idb.onerror = (event) => reject("Error in opening IndexedDB!");
|
||
idb.onsuccess = (event) => resolve(event.target.result);
|
||
});
|
||
},
|
||
|
||
deleteDB: function (dbName = this.defaultDB) {
|
||
return new Promise((resolve, reject) => {
|
||
var deleteReq = indexedDB.deleteDatabase(dbName);
|
||
deleteReq.onerror = (event) => reject("Error deleting database!");
|
||
deleteReq.onsuccess = (event) =>
|
||
resolve("Database deleted successfully");
|
||
});
|
||
},
|
||
|
||
writeData: function (
|
||
obsName,
|
||
data,
|
||
key = false,
|
||
dbName = this.defaultDB
|
||
) {
|
||
return new Promise((resolve, reject) => {
|
||
this.openDB(dbName)
|
||
.then((db) => {
|
||
var obs = db
|
||
.transaction(obsName, "readwrite")
|
||
.objectStore(obsName);
|
||
let writeReq = key ? obs.put(data, key) : obs.put(data);
|
||
writeReq.onsuccess = (evt) => resolve(`Write data Successful`);
|
||
writeReq.onerror = (evt) =>
|
||
reject(
|
||
`Write data unsuccessful [${evt.target.error.name}] ${evt.target.error.message}`
|
||
);
|
||
db.close();
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
addData: function (obsName, data, key = false, dbName = this.defaultDB) {
|
||
return new Promise((resolve, reject) => {
|
||
this.openDB(dbName)
|
||
.then((db) => {
|
||
var obs = db
|
||
.transaction(obsName, "readwrite")
|
||
.objectStore(obsName);
|
||
let addReq = key ? obs.add(data, key) : obs.add(data);
|
||
addReq.onsuccess = (evt) => resolve(`Add data successful`);
|
||
addReq.onerror = (evt) =>
|
||
reject(
|
||
`Add data unsuccessful [${evt.target.error.name}] ${evt.target.error.message}`
|
||
);
|
||
db.close();
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
removeData: function (obsName, key, dbName = this.defaultDB) {
|
||
return new Promise((resolve, reject) => {
|
||
this.openDB(dbName)
|
||
.then((db) => {
|
||
var obs = db
|
||
.transaction(obsName, "readwrite")
|
||
.objectStore(obsName);
|
||
let delReq = obs.delete(key);
|
||
delReq.onsuccess = (evt) => resolve(`Removed Data ${key}`);
|
||
delReq.onerror = (evt) =>
|
||
reject(
|
||
`Remove data unsuccessful [${evt.target.error.name}] ${evt.target.error.message}`
|
||
);
|
||
db.close();
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
clearData: function (obsName, dbName = this.defaultDB) {
|
||
return new Promise((resolve, reject) => {
|
||
this.openDB(dbName)
|
||
.then((db) => {
|
||
var obs = db
|
||
.transaction(obsName, "readwrite")
|
||
.objectStore(obsName);
|
||
let clearReq = obs.clear();
|
||
clearReq.onsuccess = (evt) => resolve(`Clear data Successful`);
|
||
clearReq.onerror = (evt) => reject(`Clear data Unsuccessful`);
|
||
db.close();
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
readData: function (obsName, key, dbName = this.defaultDB) {
|
||
return new Promise((resolve, reject) => {
|
||
this.openDB(dbName)
|
||
.then((db) => {
|
||
var obs = db
|
||
.transaction(obsName, "readonly")
|
||
.objectStore(obsName);
|
||
let getReq = obs.get(key);
|
||
getReq.onsuccess = (evt) => resolve(evt.target.result);
|
||
getReq.onerror = (evt) =>
|
||
reject(
|
||
`Read data unsuccessful [${evt.target.error.name}] ${evt.target.error.message}`
|
||
);
|
||
db.close();
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
readAllData: function (obsName, dbName = this.defaultDB) {
|
||
return new Promise((resolve, reject) => {
|
||
this.openDB(dbName)
|
||
.then((db) => {
|
||
var obs = db
|
||
.transaction(obsName, "readonly")
|
||
.objectStore(obsName);
|
||
var tmpResult = {};
|
||
let curReq = obs.openCursor();
|
||
curReq.onsuccess = (evt) => {
|
||
var cursor = evt.target.result;
|
||
if (cursor) {
|
||
tmpResult[cursor.primaryKey] = cursor.value;
|
||
cursor.continue();
|
||
} else resolve(tmpResult);
|
||
};
|
||
curReq.onerror = (evt) =>
|
||
reject(
|
||
`Read-All data unsuccessful [${evt.target.error.name}] ${evt.target.error.message}`
|
||
);
|
||
db.close();
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
searchData: function (obsName, patternEval, dbName = this.defaultDB) {
|
||
return new Promise((resolve, reject) => {
|
||
this.openDB(dbName)
|
||
.then((db) => {
|
||
var obs = db
|
||
.transaction(obsName, "readonly")
|
||
.objectStore(obsName);
|
||
var filteredResult = {};
|
||
let curReq = obs.openCursor();
|
||
curReq.onsuccess = (evt) => {
|
||
var cursor = evt.target.result;
|
||
if (cursor) {
|
||
if (patternEval(cursor.primaryKey, cursor.value))
|
||
filteredResult[cursor.primaryKey] = cursor.value;
|
||
cursor.continue();
|
||
} else resolve(filteredResult);
|
||
};
|
||
curReq.onerror = (evt) =>
|
||
reject(
|
||
`Search unsuccessful [${evt.target.error.name}] ${evt.target.error.message}`
|
||
);
|
||
db.close();
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
};
|
||
</script>
|
||
|
||
<script id="floCloudAPI">
|
||
/* FLO Cloud operations to send/request application data*/
|
||
const floCloudAPI = {
|
||
util: {
|
||
lastCommit: {},
|
||
|
||
resetData: function (dataSet) {
|
||
try {
|
||
dataSet = JSON.parse(dataSet);
|
||
console.log(dataSet);
|
||
for (vc in dataSet) {
|
||
var message = dataSet[vc].message;
|
||
if (message.reset) {
|
||
floGlobals.appObjects[message.object] = message.reset;
|
||
floGlobals.vectorClock[message.object] = vc;
|
||
compactIDB.writeData(
|
||
"appObjects",
|
||
floGlobals.appObjects[message.object],
|
||
message.object
|
||
);
|
||
compactIDB.writeData("vectorClock", vc, message.object);
|
||
}
|
||
}
|
||
} catch (error) {
|
||
console.error(error);
|
||
}
|
||
},
|
||
|
||
updateData: function (dataSet) {
|
||
try {
|
||
dataSet = JSON.parse(dataSet);
|
||
console.log(dataSet);
|
||
for (vc in dataSet) {
|
||
var message = dataSet[vc].message;
|
||
if (message.diff) {
|
||
floGlobals.appObjects[message.object] = mergeDifference(
|
||
floGlobals.appObjects[message.object],
|
||
message.diff
|
||
);
|
||
compactIDB.writeData(
|
||
"appObjects",
|
||
floGlobals.appObjects[message.object],
|
||
message.object
|
||
);
|
||
}
|
||
floGlobals.vectorClock[message.object] = vc;
|
||
compactIDB.writeData("vectorClock", vc, message.object);
|
||
}
|
||
} catch (error) {
|
||
console.error(error);
|
||
}
|
||
},
|
||
|
||
storeGeneralData: function (filterStr, dataSet) {
|
||
try {
|
||
dataSet = JSON.parse(dataSet);
|
||
console.log(dataSet);
|
||
if (!Array.isArray(floGlobals.generalData[filterStr]))
|
||
floGlobals.generalData[filterStr] = [];
|
||
for (vc in dataSet) {
|
||
floGlobals.generalData[filterStr].push({
|
||
sender: dataSet[vc].senderID,
|
||
vectorClock: vc,
|
||
message: dataSet[vc].message,
|
||
});
|
||
compactIDB.writeData(
|
||
"generalData",
|
||
floGlobals.generalData[filterStr],
|
||
filterStr
|
||
);
|
||
floGlobals.generalVC[filterStr] = vc;
|
||
compactIDB.writeData("generalVC", vc, filterStr);
|
||
}
|
||
} catch (error) {
|
||
console.error(error);
|
||
}
|
||
},
|
||
},
|
||
|
||
//send Any message to supernode cloud storage
|
||
sendApplicationData: function (message, type, options = {}) {
|
||
return new Promise((resolve, reject) => {
|
||
var data = {
|
||
senderID: myFloID,
|
||
receiverID: options.receiverID || floGlobals.adminID,
|
||
pubKey: floCrypto.getPubKeyHex(myPrivKey),
|
||
message: message,
|
||
sign: floCrypto.signData(JSON.stringify(message), myPrivKey),
|
||
application: options.application || floGlobals.application,
|
||
type: type,
|
||
comment: options.comment || "",
|
||
};
|
||
floSupernode
|
||
.sendData(JSON.stringify(data), data.receiverID)
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
//request any data from supernode cloud
|
||
requestApplicationData: function (options = {}) {
|
||
return new Promise((resolve, reject) => {
|
||
var request = {
|
||
receiverID: options.receiverID || floGlobals.adminID,
|
||
senderIDs: options.senderIDs || undefined,
|
||
application: options.application || floGlobals.application,
|
||
type: options.type || undefined,
|
||
comment: options.comment || undefined,
|
||
lowerVectorClock: options.lowerVectorClock || undefined,
|
||
upperVectorClock: options.upperVectorClock || undefined,
|
||
atVectorClock: options.atVectorClock || undefined,
|
||
mostRecent: options.mostRecent || undefined,
|
||
};
|
||
floSupernode
|
||
.requestData(JSON.stringify(request), request.receiverID)
|
||
.then((response) => resolve(response))
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
//send General Data
|
||
sendGeneralData: function (message, type, options = {}) {
|
||
return new Promise((resolve, reject) => {
|
||
this.sendApplicationData(message, type, options)
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
//request General Data
|
||
requestGeneralData: function (type, options = {}) {
|
||
return new Promise((resolve, reject) => {
|
||
var filterStr = JSON.stringify({
|
||
application: options.application || floGlobals.application,
|
||
type: type,
|
||
comment: options.comment,
|
||
});
|
||
options.type = type;
|
||
options.lowerVectorClock =
|
||
options.lowerVectorClock || floGlobals.generalVC[filterStr] + 1;
|
||
this.requestApplicationData(options)
|
||
.then((dataSet) => {
|
||
this.util.storeGeneralData(filterStr, dataSet);
|
||
resolve("General Data Updated");
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
//request an object data from supernode cloud
|
||
requestObjectData: function (objectName, options = {}) {
|
||
return new Promise((resolve, reject) => {
|
||
var request = {
|
||
receiverID: options.receiverID || floGlobals.adminID,
|
||
senderIDs:
|
||
options.senderIDs === false
|
||
? false
|
||
: options.senderIDs || floGlobals.subAdmins,
|
||
application: options.application || floGlobals.application,
|
||
comment: options.comment,
|
||
type: `${objectName}@Reset`,
|
||
lowerVectorClock: floGlobals.vectorClock[objectName] + 1,
|
||
mostRecent: true,
|
||
};
|
||
floSupernode
|
||
.requestData(JSON.stringify(request), request.receiverID)
|
||
.then((dataSet) => {
|
||
this.util.resetData(dataSet);
|
||
request.type = `${objectName}@Update`;
|
||
request.lowerVectorClock = floGlobals.vectorClock[objectName] + 1;
|
||
request.mostRecent = false;
|
||
floSupernode
|
||
.requestData(JSON.stringify(request), request.receiverID)
|
||
.then((dataSet) => {
|
||
this.util.updateData(dataSet);
|
||
this.util.lastCommit[objectName] = JSON.stringify(
|
||
floGlobals.appObjects[objectName]
|
||
);
|
||
resolve("Object Data Updated");
|
||
})
|
||
.catch((error) => reject(error));
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
//reset or initialize an object and send it to cloud
|
||
resetObjectData: function (objectName, options = {}) {
|
||
return new Promise((resolve, reject) => {
|
||
var message = {
|
||
division: options.division || floGlobals.division,
|
||
object: objectName,
|
||
vectorClock: floGlobals.vectorClock[objectName],
|
||
reset: floGlobals.appObjects[objectName],
|
||
};
|
||
this.sendApplicationData(message, `${objectName}@Reset`, options)
|
||
.then((result) => {
|
||
this.util.lastCommit[objectName] = JSON.stringify(
|
||
floGlobals.appObjects[objectName]
|
||
);
|
||
resolve(result);
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
//update the diff and send it to cloud
|
||
updateObjectData: function (objectName, options = {}) {
|
||
return new Promise((resolve, reject) => {
|
||
var message = {
|
||
division: options.division || floGlobals.division,
|
||
object: objectName,
|
||
vectorClock: floGlobals.vectorClock[objectName],
|
||
diff: findDifference(
|
||
JSON.parse(this.util.lastCommit[objectName]),
|
||
floGlobals.appObjects[objectName]
|
||
),
|
||
};
|
||
this.sendApplicationData(message, `${objectName}@Update`, options)
|
||
.then((result) => {
|
||
this.util.lastCommit[objectName] = JSON.stringify(
|
||
floGlobals.appObjects[objectName]
|
||
);
|
||
resolve(result);
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
};
|
||
</script>
|
||
|
||
<script id="floDapps">
|
||
/* General functions for FLO Dapps*/
|
||
const floDapps = {
|
||
util: {
|
||
appObs: {},
|
||
|
||
initIndexedDB: function () {
|
||
return new Promise((resolve, reject) => {
|
||
var obj = {
|
||
//general
|
||
lastTx: {},
|
||
//supernode (cloud list)
|
||
supernodes: {
|
||
indexes: {
|
||
uri: null,
|
||
pubKey: null,
|
||
},
|
||
},
|
||
//login credentials
|
||
credentials: {},
|
||
//for Dapps
|
||
subAdmins: {},
|
||
appObjects: {},
|
||
vectorClock: {},
|
||
generalData: {},
|
||
generalVC: {},
|
||
};
|
||
//add other given objectStores
|
||
for (o in this.appObs) if (!(o in obj)) obj[o] = this.appObs[o];
|
||
compactIDB
|
||
.initDB(floGlobals.application, obj)
|
||
.then((result) => {
|
||
resolve("IndexedDB App Storage Initated Successfully");
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
startUpFunctions: {
|
||
readSupernodeListFromAPI: function () {
|
||
return new Promise((resolve, reject) => {
|
||
compactIDB
|
||
.readData("lastTx", floGlobals.SNStorageID)
|
||
.then((lastTx) => {
|
||
floBlockchainAPI
|
||
.readData(floGlobals.SNStorageID, {
|
||
ignoreOld: lastTx,
|
||
sentOnly: true,
|
||
pattern: "SuperNodeStorage",
|
||
})
|
||
.then((result) => {
|
||
for (var i = result.data.length - 1; i >= 0; i--) {
|
||
var content = JSON.parse(
|
||
result.data[i]
|
||
).SuperNodeStorage;
|
||
for (sn in content.removeNodes)
|
||
compactIDB.removeData("supernodes", sn);
|
||
for (sn in content.addNodes)
|
||
compactIDB.writeData(
|
||
"supernodes",
|
||
content.addNodes[sn],
|
||
sn
|
||
);
|
||
}
|
||
compactIDB.writeData(
|
||
"lastTx",
|
||
result.totalTxs,
|
||
floGlobals.SNStorageID
|
||
);
|
||
compactIDB.readAllData("supernodes").then((result) => {
|
||
floGlobals.supernodes = result;
|
||
|
||
floSupernode.kBucket
|
||
.launch(
|
||
Object.keys(floGlobals.supernodes),
|
||
floGlobals.SNStorageID
|
||
)
|
||
.then((result) =>
|
||
resolve("Loaded Supernode list\n" + result)
|
||
);
|
||
});
|
||
});
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
readSubAdminListFromAPI: function () {
|
||
return new Promise((resolve, reject) => {
|
||
compactIDB
|
||
.readData("lastTx", floGlobals.adminID)
|
||
.then((lastTx) => {
|
||
floBlockchainAPI
|
||
.readData(floGlobals.adminID, {
|
||
ignoreOld: lastTx,
|
||
sentOnly: true,
|
||
pattern: floGlobals.application,
|
||
})
|
||
.then((result) => {
|
||
for (var i = result.data.length - 1; i >= 0; i--) {
|
||
var content = JSON.parse(result.data[i])[
|
||
floGlobals.application
|
||
];
|
||
if (Array.isArray(content.removeSubAdmin))
|
||
for (
|
||
var j = 0;
|
||
j < content.removeSubAdmin.length;
|
||
j++
|
||
)
|
||
compactIDB.removeData(
|
||
"subAdmins",
|
||
content.removeSubAdmin[j]
|
||
);
|
||
if (Array.isArray(content.addSubAdmin))
|
||
for (var k = 0; k < content.addSubAdmin.length; k++)
|
||
compactIDB.writeData(
|
||
"subAdmins",
|
||
true,
|
||
content.addSubAdmin[k]
|
||
);
|
||
}
|
||
compactIDB.writeData(
|
||
"lastTx",
|
||
result.totalTxs,
|
||
floGlobals.adminID
|
||
);
|
||
compactIDB.readAllData("subAdmins").then((result) => {
|
||
floGlobals.subAdmins = Object.keys(result);
|
||
resolve("Read subAdmins from blockchain");
|
||
});
|
||
});
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
loadDataFromIDB: function () {
|
||
return new Promise((resolve, reject) => {
|
||
var loadData = [
|
||
"appObjects",
|
||
"vectorClock",
|
||
"generalData",
|
||
"generalVC",
|
||
];
|
||
var promises = [];
|
||
for (var i = 0; i < loadData.length; i++)
|
||
promises[i] = compactIDB.readAllData(loadData[i]);
|
||
Promise.all(promises)
|
||
.then((results) => {
|
||
for (var i = 0; i < loadData.length; i++)
|
||
floGlobals[loadData[i]] = results[i];
|
||
resolve("Loaded Data from IDB");
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
getCredentials: function () {
|
||
const defaultInput = function (type) {
|
||
return new Promise((resolve, reject) => {
|
||
let inputVal = prompt(`Enter ${type}: `);
|
||
if (inputVal === null) reject(null);
|
||
else resolve(inputVal);
|
||
});
|
||
};
|
||
|
||
const inputFn = this.getCredentials.privKeyInput || defaultInput;
|
||
|
||
const readSharesFromIDB = function (indexArr) {
|
||
return new Promise((resolve, reject) => {
|
||
var promises = [];
|
||
for (var i = 0; i < indexArr.length; i++)
|
||
promises.push(
|
||
compactIDB.readData("credentials", indexArr[i])
|
||
);
|
||
Promise.all(promises)
|
||
.then((shares) => {
|
||
var secret = floCrypto.retrieveShamirSecret(shares);
|
||
if (secret) resolve(secret);
|
||
else reject("Shares are insufficient or incorrect");
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
};
|
||
|
||
const writeSharesToIDB = function (
|
||
shares,
|
||
i = 0,
|
||
resultIndexes = []
|
||
) {
|
||
return new Promise((resolve, reject) => {
|
||
if (i >= shares.length) return resolve(resultIndexes);
|
||
var n = floCrypto.randInt(0, 100000);
|
||
compactIDB
|
||
.addData("credentials", shares[i], n)
|
||
.then((res) => {
|
||
resultIndexes.push(n);
|
||
writeSharesToIDB(shares, i + 1, resultIndexes).then(
|
||
(result) => resolve(result)
|
||
);
|
||
})
|
||
.catch((error) => {
|
||
writeSharesToIDB(shares, i, resultIndexes).then((result) =>
|
||
resolve(result)
|
||
);
|
||
});
|
||
});
|
||
};
|
||
|
||
const getPrivateKeyCredentials = function () {
|
||
return new Promise((resolve, reject) => {
|
||
var indexArr = localStorage.getItem(
|
||
`${floGlobals.application}#privKey`
|
||
);
|
||
if (indexArr) {
|
||
readSharesFromIDB(JSON.parse(indexArr))
|
||
.then((result) => resolve(result))
|
||
.catch((error) => reject(error));
|
||
} else {
|
||
var privKey;
|
||
inputFn("PRIVATE_KEY")
|
||
.then((result) => {
|
||
try {
|
||
if (!result) return reject("Empty Private Key");
|
||
var floID = floCrypto.getFloIDfromPubkeyHex(
|
||
floCrypto.getPubKeyHex(result)
|
||
);
|
||
privKey = result;
|
||
} catch (error) {
|
||
console.error(error);
|
||
return reject("Invalid Private Key");
|
||
}
|
||
})
|
||
.catch((error) => {
|
||
console.log(error, "Generating Random Keys");
|
||
privKey = floCrypto.generateNewID().privKey;
|
||
})
|
||
.finally((_) => {
|
||
var threshold = floCrypto.randInt(10, 20);
|
||
writeSharesToIDB(
|
||
floCrypto.createShamirsSecretShares(
|
||
privKey,
|
||
threshold,
|
||
threshold
|
||
)
|
||
)
|
||
.then((resultIndexes) => {
|
||
//store index keys in localStorage
|
||
localStorage.setItem(
|
||
`${floGlobals.application}#privKey`,
|
||
JSON.stringify(resultIndexes)
|
||
);
|
||
//also add a dummy privatekey to the IDB
|
||
var randomPrivKey = floCrypto.generateNewID().privKey;
|
||
var randomThreshold = floCrypto.randInt(10, 20);
|
||
writeSharesToIDB(
|
||
floCrypto.createShamirsSecretShares(
|
||
randomPrivKey,
|
||
randomThreshold,
|
||
randomThreshold
|
||
)
|
||
);
|
||
//resolve private Key
|
||
resolve(privKey);
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
}
|
||
});
|
||
};
|
||
|
||
const checkIfPinRequired = function (key) {
|
||
return new Promise((resolve, reject) => {
|
||
if (key.length == 52) resolve(key);
|
||
else {
|
||
inputFn("PIN/Password")
|
||
.then((pwd) => {
|
||
try {
|
||
let privKey = Crypto.AES.decrypt(key, pwd);
|
||
resolve(privKey);
|
||
} catch (error) {
|
||
reject("Access Denied: Incorrect PIN/Password");
|
||
}
|
||
})
|
||
.catch((error) =>
|
||
reject("Access Denied: PIN/Password required")
|
||
);
|
||
}
|
||
});
|
||
};
|
||
|
||
return new Promise((resolve, reject) => {
|
||
getPrivateKeyCredentials()
|
||
.then((key) => {
|
||
checkIfPinRequired(key)
|
||
.then((privKey) => {
|
||
try {
|
||
myPrivKey = privKey;
|
||
myPubKey = floCrypto.getPubKeyHex(myPrivKey);
|
||
myFloID = floCrypto.getFloIDfromPubkeyHex(myPubKey);
|
||
resolve("Login Credentials loaded successful");
|
||
} catch (error) {
|
||
reject("Corrupted Private Key");
|
||
}
|
||
})
|
||
.catch((error) => reject(error));
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
},
|
||
|
||
callStartUpFunction: function (fname) {
|
||
return new Promise((resolve, reject) => {
|
||
this.startUpFunctions[fname]()
|
||
.then((result) => {
|
||
this.callStartUpFunction.completed += 1;
|
||
reactor.dispatchEvent(
|
||
"startUpSuccessLog",
|
||
`${result}\nCompleted ${this.callStartUpFunction.completed}/${this.callStartUpFunction.total} Startup functions`
|
||
);
|
||
resolve(true);
|
||
})
|
||
.catch((error) => {
|
||
this.callStartUpFunction.failed += 1;
|
||
reactor.dispatchEvent(
|
||
"startUpErrorLog",
|
||
`${error}\nFailed ${this.callStartUpFunction.failed}/${this.callStartUpFunction.total} Startup functions`
|
||
);
|
||
reject(false);
|
||
});
|
||
});
|
||
},
|
||
|
||
getFilterString: function (type, options = {}) {
|
||
var filterStr = JSON.stringify({
|
||
application: options.application || floGlobals.application,
|
||
type: type,
|
||
comment: options.comment,
|
||
});
|
||
return filterStr;
|
||
},
|
||
},
|
||
|
||
launchStartUp: function () {
|
||
return new Promise((resolve, reject) => {
|
||
this.util.initIndexedDB().then((log) => {
|
||
console.log(log);
|
||
this.util.callStartUpFunction.total = Object.keys(
|
||
this.util.startUpFunctions
|
||
).length;
|
||
this.util.callStartUpFunction.completed = 0;
|
||
this.util.callStartUpFunction.failed = 0;
|
||
var promises = [];
|
||
for (fn in this.util.startUpFunctions)
|
||
promises.push(this.util.callStartUpFunction(fn));
|
||
Promise.all(promises)
|
||
.then((results) => resolve("App Startup finished successful"))
|
||
.catch((errors) => reject("App StartUp failed"));
|
||
});
|
||
});
|
||
},
|
||
|
||
addStartUpFunction: function (fname, fn) {
|
||
if (fname in this.util.startUpFunctions)
|
||
throw `Function ${fname} already defined`;
|
||
this.util.startUpFunctions[fname] = fn;
|
||
},
|
||
|
||
setCustomPrivKeyInput: function (customFn) {
|
||
this.util.startUpFunctions.getCredentials.privKeyInput = customFn;
|
||
},
|
||
|
||
setAppObjectStores: function (appObs) {
|
||
this.util.appObs = appObs;
|
||
},
|
||
|
||
manageSubAdmins(adminPrivKey, addList, rmList) {
|
||
return new Promise((resolve, reject) => {
|
||
if (!Array.isArray(addList)) addList = undefined;
|
||
if (!Array.isArray(rmList)) rmList = undefined;
|
||
var floData = {
|
||
[floGlobals.application]: {
|
||
addSubAdmin: addList,
|
||
removeSubAdmin: rmList,
|
||
},
|
||
};
|
||
var floID = floCrypto.getFloIDfromPubkeyHex(
|
||
floCrypto.getPubKeyHex(adminPrivKey)
|
||
);
|
||
if (floID != floGlobals.adminID)
|
||
reject("Access Denied for Admin privilege");
|
||
else
|
||
floBlockchainAPI
|
||
.writeData(floID, JSON.stringify(floData), adminPrivKey)
|
||
.then((result) => resolve(["Updated SubAdmin List", result]))
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
clearCredentials: function () {
|
||
return new Promise((resolve, reject) => {
|
||
compactIDB
|
||
.clearData("credentials")
|
||
.then((result) => {
|
||
localStorage.removeItem(`${floGlobals.application}#privKey`);
|
||
myPrivKey = myPubKey = myFloID = undefined;
|
||
resolve("privKey credentials deleted!");
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
securePrivKey: function (pwd) {
|
||
return new Promise((resolve, reject) => {
|
||
let indexArr = localStorage.getItem(
|
||
`${floGlobals.application}#privKey`
|
||
);
|
||
if (!indexArr) return reject("PrivKey not found");
|
||
indexArr = JSON.parse(indexArr);
|
||
let encryptedKey = Crypto.AES.encrypt(myPrivKey, pwd);
|
||
let threshold = indexArr.length;
|
||
let shares = floCrypto.createShamirsSecretShares(
|
||
encryptedKey,
|
||
threshold,
|
||
threshold
|
||
);
|
||
let promises = [];
|
||
for (var i = 0; i < threshold; i++)
|
||
promises.push(
|
||
compactIDB.writeData(
|
||
"credentials",
|
||
shares[i],
|
||
indexArr[i],
|
||
floGlobals.application
|
||
)
|
||
);
|
||
Promise.all(promises)
|
||
.then((results) => resolve("Private Key Secured"))
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
objectDataMapper: function (object, path, data) {
|
||
var resObject = JSON.parse(JSON.stringify(object));
|
||
var pos = resObject;
|
||
path.forEach((p) => (pos = pos[p]));
|
||
if (Array.isArray(pos)) {
|
||
pos.push(data);
|
||
return resObject;
|
||
} else throw "Path is not an Array";
|
||
},
|
||
|
||
getNextGeneralData: function (type, vectorClock, options = {}) {
|
||
var filter = this.util.getFilterString(type, options);
|
||
var filteredResult = [];
|
||
for (var i = 0; i < floGlobals.generalData[filter].length; i++)
|
||
if (floGlobals.generalData[filter][i].vectorClock > vectorClock)
|
||
filteredResult.push(floGlobals.generalData[filter][i]);
|
||
return filteredResult;
|
||
},
|
||
};
|
||
|
||
reactor.registerEvent("startUpSuccessLog");
|
||
reactor.addEventListener("startUpSuccessLog", (log) => console.log(log));
|
||
|
||
reactor.registerEvent("startUpErrorLog");
|
||
reactor.addEventListener("startUpErrorLog", (log) => console.error(log));
|
||
</script>
|
||
|
||
<script id="webAppClient">
|
||
//Add this to the client script to request data from floWebappServer
|
||
const webAppClient = {
|
||
util: {
|
||
requestData(request = []) {
|
||
return new Promise((resolve, reject) => {
|
||
var websocket = new WebSocket(
|
||
"wss://" + floGlobals.webAppURL + "/ws"
|
||
);
|
||
websocket.onmessage = (evt) => {
|
||
if (evt.data == "$+") {
|
||
websocket.send(`?${JSON.stringify(request)}`);
|
||
} else if (evt.data == "$-") {
|
||
reject(`webApp not available`);
|
||
websocket.close();
|
||
} else {
|
||
resolve(JSON.parse(evt.data));
|
||
websocket.close();
|
||
}
|
||
};
|
||
websocket.onerror = (evt) => {
|
||
reject(`webApp server not found`);
|
||
};
|
||
});
|
||
},
|
||
},
|
||
|
||
requestGeneralData: function (type, vectorClock = "0") {
|
||
if (typeof vectorClock !== "string") vectorClock.toString();
|
||
return new Promise((resolve, reject) => {
|
||
this.util
|
||
.requestData([
|
||
floGlobals.application,
|
||
"generalData",
|
||
type,
|
||
vectorClock,
|
||
])
|
||
.then((result) => {
|
||
if (result[0]) resolve(result[1]);
|
||
else reject(result[1]);
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
requestObjectData: function (keyPath) {
|
||
return new Promise((resolve, reject) => {
|
||
this.util
|
||
.requestData([floGlobals.application, "appObjects"].concat(keyPath))
|
||
.then((result) => {
|
||
if (result[0]) resolve(result[1]);
|
||
else reject(result[1]);
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
|
||
requestSubAdminList: function () {
|
||
return new Promise((resolve, reject) => {
|
||
this.util
|
||
.requestData([floGlobals.application, "subAdmins"])
|
||
.then((result) => {
|
||
if (result[0]) resolve(result[1]);
|
||
else reject(result[1]);
|
||
})
|
||
.catch((error) => reject(error));
|
||
});
|
||
},
|
||
};
|
||
</script>
|
||
</body>
|
||
|
||
</html> |