mirror of
https://github.com/naruxde/revpicommander.git
synced 2025-11-09 08:58:04 +01:00
Compare commits
28 Commits
0.9.10rc2
...
debian/0.1
| Author | SHA1 | Date | |
|---|---|---|---|
| b252aec126 | |||
| 3d6e9877d7 | |||
| c10a50fa56 | |||
| e4d18f7b4a | |||
| efd27cc96c | |||
| 0494ce53ad | |||
| c56c33f4e5 | |||
| e95bf70993 | |||
| 6ee53595e2 | |||
| a0b309ade0 | |||
| be03bbe6f3 | |||
| 92abe46152 | |||
| 27915e5a58 | |||
| a8e4638ab9 | |||
| 7262ad7e1a | |||
| 95d3094623 | |||
| a96d7d4269 | |||
| 43d4a97edd | |||
| 3eaae9e462 | |||
| 7a1038a5ab | |||
| bfaa07aeb4 | |||
| 43d03cc254 | |||
| d203958b5f | |||
| 9202aedf7f | |||
| 8e2b8311ef | |||
| 15058feb7e | |||
| d14b6dccdd | |||
| feb55fe2b5 |
2
.idea/misc.xml
generated
2
.idea/misc.xml
generated
@@ -6,7 +6,7 @@
|
|||||||
<component name="ProjectPlainTextFileTypeManager">
|
<component name="ProjectPlainTextFileTypeManager">
|
||||||
<file url="file://$PROJECT_DIR$/src/revpicommander/locale/revpicommander_de.ts" />
|
<file url="file://$PROJECT_DIR$/src/revpicommander/locale/revpicommander_de.ts" />
|
||||||
</component>
|
</component>
|
||||||
<component name="ProjectRootManager" version="2" project-jdk-name="Python 3.9 (revpicommander)" project-jdk-type="Python SDK" />
|
<component name="ProjectRootManager" version="2" project-jdk-name="Python 3.11 (revpicommander)" project-jdk-type="Python SDK" />
|
||||||
<component name="PythonCompatibilityInspectionAdvertiser">
|
<component name="PythonCompatibilityInspectionAdvertiser">
|
||||||
<option name="version" value="3" />
|
<option name="version" value="3" />
|
||||||
</component>
|
</component>
|
||||||
|
|||||||
2
.idea/revpicommander.iml
generated
2
.idea/revpicommander.iml
generated
@@ -6,7 +6,7 @@
|
|||||||
<excludeFolder url="file://$MODULE_DIR$/venv" />
|
<excludeFolder url="file://$MODULE_DIR$/venv" />
|
||||||
<excludeFolder url="file://$MODULE_DIR$/build" />
|
<excludeFolder url="file://$MODULE_DIR$/build" />
|
||||||
</content>
|
</content>
|
||||||
<orderEntry type="jdk" jdkName="Python 3.9 (revpicommander)" jdkType="Python SDK" />
|
<orderEntry type="jdk" jdkName="Python 3.11 (revpicommander)" jdkType="Python SDK" />
|
||||||
<orderEntry type="sourceFolder" forTests="false" />
|
<orderEntry type="sourceFolder" forTests="false" />
|
||||||
</component>
|
</component>
|
||||||
<component name="PackageRequirementsSettings">
|
<component name="PackageRequirementsSettings">
|
||||||
|
|||||||
2
.idea/vcs.xml
generated
2
.idea/vcs.xml
generated
@@ -3,6 +3,8 @@
|
|||||||
<component name="CommitMessageInspectionProfile">
|
<component name="CommitMessageInspectionProfile">
|
||||||
<profile version="1.0">
|
<profile version="1.0">
|
||||||
<inspection_tool class="BodyLimit" enabled="true" level="WARNING" enabled_by_default="true" />
|
<inspection_tool class="BodyLimit" enabled="true" level="WARNING" enabled_by_default="true" />
|
||||||
|
<inspection_tool class="CommitFormat" enabled="true" level="WARNING" enabled_by_default="true" />
|
||||||
|
<inspection_tool class="CommitNamingConvention" enabled="true" level="WARNING" enabled_by_default="true" />
|
||||||
<inspection_tool class="SubjectBodySeparation" enabled="true" level="WARNING" enabled_by_default="true" />
|
<inspection_tool class="SubjectBodySeparation" enabled="true" level="WARNING" enabled_by_default="true" />
|
||||||
<inspection_tool class="SubjectLimit" enabled="true" level="WARNING" enabled_by_default="true" />
|
<inspection_tool class="SubjectLimit" enabled="true" level="WARNING" enabled_by_default="true" />
|
||||||
</profile>
|
</profile>
|
||||||
|
|||||||
873
LICENSE.txt
873
LICENSE.txt
@@ -1,622 +1,281 @@
|
|||||||
GNU GENERAL PUBLIC LICENSE
|
GNU GENERAL PUBLIC LICENSE
|
||||||
Version 3, 29 June 2007
|
Version 2, June 1991
|
||||||
|
|
||||||
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
|
||||||
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
Everyone is permitted to copy and distribute verbatim copies
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
of this license document, but changing it is not allowed.
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
Preamble
|
Preamble
|
||||||
|
|
||||||
The GNU General Public License is a free, copyleft license for
|
The licenses for most software are designed to take away your
|
||||||
software and other kinds of works.
|
freedom to share and change it. By contrast, the GNU General Public
|
||||||
|
License is intended to guarantee your freedom to share and change free
|
||||||
The licenses for most software and other practical works are designed
|
software--to make sure the software is free for all its users. This
|
||||||
to take away your freedom to share and change the works. By contrast,
|
General Public License applies to most of the Free Software
|
||||||
the GNU General Public License is intended to guarantee your freedom to
|
Foundation's software and to any other program whose authors commit to
|
||||||
share and change all versions of a program--to make sure it remains free
|
using it. (Some other Free Software Foundation software is covered by
|
||||||
software for all its users. We, the Free Software Foundation, use the
|
the GNU Lesser General Public License instead.) You can apply it to
|
||||||
GNU General Public License for most of our software; it applies also to
|
|
||||||
any other work released this way by its authors. You can apply it to
|
|
||||||
your programs, too.
|
your programs, too.
|
||||||
|
|
||||||
When we speak of free software, we are referring to freedom, not
|
When we speak of free software, we are referring to freedom, not
|
||||||
price. Our General Public Licenses are designed to make sure that you
|
price. Our General Public Licenses are designed to make sure that you
|
||||||
have the freedom to distribute copies of free software (and charge for
|
have the freedom to distribute copies of free software (and charge for
|
||||||
them if you wish), that you receive source code or can get it if you
|
this service if you wish), that you receive source code or can get it
|
||||||
want it, that you can change the software or use pieces of it in new
|
if you want it, that you can change the software or use pieces of it
|
||||||
free programs, and that you know you can do these things.
|
in new free programs; and that you know you can do these things.
|
||||||
|
|
||||||
To protect your rights, we need to prevent others from denying you
|
To protect your rights, we need to make restrictions that forbid
|
||||||
these rights or asking you to surrender the rights. Therefore, you have
|
anyone to deny you these rights or to ask you to surrender the rights.
|
||||||
certain responsibilities if you distribute copies of the software, or if
|
These restrictions translate to certain responsibilities for you if you
|
||||||
you modify it: responsibilities to respect the freedom of others.
|
distribute copies of the software, or if you modify it.
|
||||||
|
|
||||||
For example, if you distribute copies of such a program, whether
|
For example, if you distribute copies of such a program, whether
|
||||||
gratis or for a fee, you must pass on to the recipients the same
|
gratis or for a fee, you must give the recipients all the rights that
|
||||||
freedoms that you received. You must make sure that they, too, receive
|
you have. You must make sure that they, too, receive or can get the
|
||||||
or can get the source code. And you must show them these terms so they
|
source code. And you must show them these terms so they know their
|
||||||
know their rights.
|
rights.
|
||||||
|
|
||||||
Developers that use the GNU GPL protect your rights with two steps:
|
We protect your rights with two steps: (1) copyright the software, and
|
||||||
(1) assert copyright on the software, and (2) offer you this License
|
(2) offer you this license which gives you legal permission to copy,
|
||||||
giving you legal permission to copy, distribute and/or modify it.
|
distribute and/or modify the software.
|
||||||
|
|
||||||
For the developers' and authors' protection, the GPL clearly explains
|
Also, for each author's protection and ours, we want to make certain
|
||||||
that there is no warranty for this free software. For both users' and
|
that everyone understands that there is no warranty for this free
|
||||||
authors' sake, the GPL requires that modified versions be marked as
|
software. If the software is modified by someone else and passed on, we
|
||||||
changed, so that their problems will not be attributed erroneously to
|
want its recipients to know that what they have is not the original, so
|
||||||
authors of previous versions.
|
that any problems introduced by others will not reflect on the original
|
||||||
|
authors' reputations.
|
||||||
|
|
||||||
Some devices are designed to deny users access to install or run
|
Finally, any free program is threatened constantly by software
|
||||||
modified versions of the software inside them, although the manufacturer
|
patents. We wish to avoid the danger that redistributors of a free
|
||||||
can do so. This is fundamentally incompatible with the aim of
|
program will individually obtain patent licenses, in effect making the
|
||||||
protecting users' freedom to change the software. The systematic
|
program proprietary. To prevent this, we have made it clear that any
|
||||||
pattern of such abuse occurs in the area of products for individuals to
|
patent must be licensed for everyone's free use or not licensed at all.
|
||||||
use, which is precisely where it is most unacceptable. Therefore, we
|
|
||||||
have designed this version of the GPL to prohibit the practice for those
|
|
||||||
products. If such problems arise substantially in other domains, we
|
|
||||||
stand ready to extend this provision to those domains in future versions
|
|
||||||
of the GPL, as needed to protect the freedom of users.
|
|
||||||
|
|
||||||
Finally, every program is threatened constantly by software patents.
|
|
||||||
States should not allow patents to restrict development and use of
|
|
||||||
software on general-purpose computers, but in those that do, we wish to
|
|
||||||
avoid the special danger that patents applied to a free program could
|
|
||||||
make it effectively proprietary. To prevent this, the GPL assures that
|
|
||||||
patents cannot be used to render the program non-free.
|
|
||||||
|
|
||||||
The precise terms and conditions for copying, distribution and
|
The precise terms and conditions for copying, distribution and
|
||||||
modification follow.
|
modification follow.
|
||||||
|
|
||||||
TERMS AND CONDITIONS
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||||
0. Definitions.
|
|
||||||
|
0. This License applies to any program or other work which contains
|
||||||
"This License" refers to version 3 of the GNU General Public License.
|
a notice placed by the copyright holder saying it may be distributed
|
||||||
|
under the terms of this General Public License. The "Program", below,
|
||||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
refers to any such program or work, and a "work based on the Program"
|
||||||
works, such as semiconductor masks.
|
means either the Program or any derivative work under copyright law:
|
||||||
|
that is to say, a work containing the Program or a portion of it,
|
||||||
"The Program" refers to any copyrightable work licensed under this
|
either verbatim or with modifications and/or translated into another
|
||||||
License. Each licensee is addressed as "you". "Licensees" and
|
language. (Hereinafter, translation is included without limitation in
|
||||||
"recipients" may be individuals or organizations.
|
the term "modification".) Each licensee is addressed as "you".
|
||||||
|
|
||||||
To "modify" a work means to copy from or adapt all or part of the work
|
Activities other than copying, distribution and modification are not
|
||||||
in a fashion requiring copyright permission, other than the making of an
|
covered by this License; they are outside its scope. The act of
|
||||||
exact copy. The resulting work is called a "modified version" of the
|
running the Program is not restricted, and the output from the Program
|
||||||
earlier work or a work "based on" the earlier work.
|
is covered only if its contents constitute a work based on the
|
||||||
|
Program (independent of having been made by running the Program).
|
||||||
A "covered work" means either the unmodified Program or a work based
|
Whether that is true depends on what the Program does.
|
||||||
on the Program.
|
|
||||||
|
1. You may copy and distribute verbatim copies of the Program's
|
||||||
To "propagate" a work means to do anything with it that, without
|
source code as you receive it, in any medium, provided that you
|
||||||
permission, would make you directly or secondarily liable for
|
conspicuously and appropriately publish on each copy an appropriate
|
||||||
infringement under applicable copyright law, except executing it on a
|
copyright notice and disclaimer of warranty; keep intact all the
|
||||||
computer or modifying a private copy. Propagation includes copying,
|
notices that refer to this License and to the absence of any warranty;
|
||||||
distribution (with or without modification), making available to the
|
and give any other recipients of the Program a copy of this License
|
||||||
public, and in some countries other activities as well.
|
along with the Program.
|
||||||
|
|
||||||
To "convey" a work means any kind of propagation that enables other
|
You may charge a fee for the physical act of transferring a copy, and
|
||||||
parties to make or receive copies. Mere interaction with a user through
|
you may at your option offer warranty protection in exchange for a fee.
|
||||||
a computer network, with no transfer of a copy, is not conveying.
|
|
||||||
|
2. You may modify your copy or copies of the Program or any portion
|
||||||
An interactive user interface displays "Appropriate Legal Notices"
|
of it, thus forming a work based on the Program, and copy and
|
||||||
to the extent that it includes a convenient and prominently visible
|
distribute such modifications or work under the terms of Section 1
|
||||||
feature that (1) displays an appropriate copyright notice, and (2)
|
above, provided that you also meet all of these conditions:
|
||||||
tells the user that there is no warranty for the work (except to the
|
|
||||||
extent that warranties are provided), that licensees may convey the
|
a) You must cause the modified files to carry prominent notices
|
||||||
work under this License, and how to view a copy of this License. If
|
stating that you changed the files and the date of any change.
|
||||||
the interface presents a list of user commands or options, such as a
|
|
||||||
menu, a prominent item in the list meets this criterion.
|
b) You must cause any work that you distribute or publish, that in
|
||||||
|
whole or in part contains or is derived from the Program or any
|
||||||
1. Source Code.
|
part thereof, to be licensed as a whole at no charge to all third
|
||||||
|
parties under the terms of this License.
|
||||||
The "source code" for a work means the preferred form of the work
|
|
||||||
for making modifications to it. "Object code" means any non-source
|
c) If the modified program normally reads commands interactively
|
||||||
form of a work.
|
when run, you must cause it, when started running for such
|
||||||
|
interactive use in the most ordinary way, to print or display an
|
||||||
A "Standard Interface" means an interface that either is an official
|
announcement including an appropriate copyright notice and a
|
||||||
standard defined by a recognized standards body, or, in the case of
|
notice that there is no warranty (or else, saying that you provide
|
||||||
interfaces specified for a particular programming language, one that
|
a warranty) and that users may redistribute the program under
|
||||||
is widely used among developers working in that language.
|
these conditions, and telling the user how to view a copy of this
|
||||||
|
License. (Exception: if the Program itself is interactive but
|
||||||
The "System Libraries" of an executable work include anything, other
|
does not normally print such an announcement, your work based on
|
||||||
than the work as a whole, that (a) is included in the normal form of
|
the Program is not required to print an announcement.)
|
||||||
packaging a Major Component, but which is not part of that Major
|
|
||||||
Component, and (b) serves only to enable use of the work with that
|
These requirements apply to the modified work as a whole. If
|
||||||
Major Component, or to implement a Standard Interface for which an
|
identifiable sections of that work are not derived from the Program,
|
||||||
implementation is available to the public in source code form. A
|
and can be reasonably considered independent and separate works in
|
||||||
"Major Component", in this context, means a major essential component
|
themselves, then this License, and its terms, do not apply to those
|
||||||
(kernel, window system, and so on) of the specific operating system
|
sections when you distribute them as separate works. But when you
|
||||||
(if any) on which the executable work runs, or a compiler used to
|
distribute the same sections as part of a whole which is a work based
|
||||||
produce the work, or an object code interpreter used to run it.
|
on the Program, the distribution of the whole must be on the terms of
|
||||||
|
this License, whose permissions for other licensees extend to the
|
||||||
The "Corresponding Source" for a work in object code form means all
|
entire whole, and thus to each and every part regardless of who wrote it.
|
||||||
the source code needed to generate, install, and (for an executable
|
|
||||||
work) run the object code and to modify the work, including scripts to
|
Thus, it is not the intent of this section to claim rights or contest
|
||||||
control those activities. However, it does not include the work's
|
your rights to work written entirely by you; rather, the intent is to
|
||||||
System Libraries, or general-purpose tools or generally available free
|
exercise the right to control the distribution of derivative or
|
||||||
programs which are used unmodified in performing those activities but
|
collective works based on the Program.
|
||||||
which are not part of the work. For example, Corresponding Source
|
|
||||||
includes interface definition files associated with source files for
|
In addition, mere aggregation of another work not based on the Program
|
||||||
the work, and the source code for shared libraries and dynamically
|
with the Program (or with a work based on the Program) on a volume of
|
||||||
linked subprograms that the work is specifically designed to require,
|
a storage or distribution medium does not bring the other work under
|
||||||
such as by intimate data communication or control flow between those
|
the scope of this License.
|
||||||
subprograms and other parts of the work.
|
|
||||||
|
3. You may copy and distribute the Program (or a work based on it,
|
||||||
The Corresponding Source need not include anything that users
|
under Section 2) in object code or executable form under the terms of
|
||||||
can regenerate automatically from other parts of the Corresponding
|
Sections 1 and 2 above provided that you also do one of the following:
|
||||||
Source.
|
|
||||||
|
a) Accompany it with the complete corresponding machine-readable
|
||||||
The Corresponding Source for a work in source code form is that
|
source code, which must be distributed under the terms of Sections
|
||||||
same work.
|
1 and 2 above on a medium customarily used for software interchange; or,
|
||||||
|
|
||||||
2. Basic Permissions.
|
b) Accompany it with a written offer, valid for at least three
|
||||||
|
years, to give any third party, for a charge no more than your
|
||||||
All rights granted under this License are granted for the term of
|
cost of physically performing source distribution, a complete
|
||||||
copyright on the Program, and are irrevocable provided the stated
|
machine-readable copy of the corresponding source code, to be
|
||||||
conditions are met. This License explicitly affirms your unlimited
|
distributed under the terms of Sections 1 and 2 above on a medium
|
||||||
permission to run the unmodified Program. The output from running a
|
customarily used for software interchange; or,
|
||||||
covered work is covered by this License only if the output, given its
|
|
||||||
content, constitutes a covered work. This License acknowledges your
|
c) Accompany it with the information you received as to the offer
|
||||||
rights of fair use or other equivalent, as provided by copyright law.
|
to distribute corresponding source code. (This alternative is
|
||||||
|
allowed only for noncommercial distribution and only if you
|
||||||
You may make, run and propagate covered works that you do not
|
received the program in object code or executable form with such
|
||||||
convey, without conditions so long as your license otherwise remains
|
an offer, in accord with Subsection b above.)
|
||||||
in force. You may convey covered works to others for the sole purpose
|
|
||||||
of having them make modifications exclusively for you, or provide you
|
The source code for a work means the preferred form of the work for
|
||||||
with facilities for running those works, provided that you comply with
|
making modifications to it. For an executable work, complete source
|
||||||
the terms of this License in conveying all material for which you do
|
code means all the source code for all modules it contains, plus any
|
||||||
not control copyright. Those thus making or running the covered works
|
associated interface definition files, plus the scripts used to
|
||||||
for you must do so exclusively on your behalf, under your direction
|
control compilation and installation of the executable. However, as a
|
||||||
and control, on terms that prohibit them from making any copies of
|
special exception, the source code distributed need not include
|
||||||
your copyrighted material outside their relationship with you.
|
anything that is normally distributed (in either source or binary
|
||||||
|
form) with the major components (compiler, kernel, and so on) of the
|
||||||
Conveying under any other circumstances is permitted solely under
|
operating system on which the executable runs, unless that component
|
||||||
the conditions stated below. Sublicensing is not allowed; section 10
|
itself accompanies the executable.
|
||||||
makes it unnecessary.
|
|
||||||
|
If distribution of executable or object code is made by offering
|
||||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
access to copy from a designated place, then offering equivalent
|
||||||
|
access to copy the source code from the same place counts as
|
||||||
No covered work shall be deemed part of an effective technological
|
distribution of the source code, even though third parties are not
|
||||||
measure under any applicable law fulfilling obligations under article
|
compelled to copy the source along with the object code.
|
||||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
|
||||||
similar laws prohibiting or restricting circumvention of such
|
4. You may not copy, modify, sublicense, or distribute the Program
|
||||||
measures.
|
except as expressly provided under this License. Any attempt
|
||||||
|
otherwise to copy, modify, sublicense or distribute the Program is
|
||||||
When you convey a covered work, you waive any legal power to forbid
|
void, and will automatically terminate your rights under this License.
|
||||||
circumvention of technological measures to the extent such circumvention
|
However, parties who have received copies, or rights, from you under
|
||||||
is effected by exercising rights under this License with respect to
|
this License will not have their licenses terminated so long as such
|
||||||
the covered work, and you disclaim any intention to limit operation or
|
parties remain in full compliance.
|
||||||
modification of the work as a means of enforcing, against the work's
|
|
||||||
users, your or third parties' legal rights to forbid circumvention of
|
5. You are not required to accept this License, since you have not
|
||||||
technological measures.
|
signed it. However, nothing else grants you permission to modify or
|
||||||
|
distribute the Program or its derivative works. These actions are
|
||||||
4. Conveying Verbatim Copies.
|
prohibited by law if you do not accept this License. Therefore, by
|
||||||
|
modifying or distributing the Program (or any work based on the
|
||||||
You may convey verbatim copies of the Program's source code as you
|
Program), you indicate your acceptance of this License to do so, and
|
||||||
receive it, in any medium, provided that you conspicuously and
|
all its terms and conditions for copying, distributing or modifying
|
||||||
appropriately publish on each copy an appropriate copyright notice;
|
the Program or works based on it.
|
||||||
keep intact all notices stating that this License and any
|
|
||||||
non-permissive terms added in accord with section 7 apply to the code;
|
6. Each time you redistribute the Program (or any work based on the
|
||||||
keep intact all notices of the absence of any warranty; and give all
|
Program), the recipient automatically receives a license from the
|
||||||
recipients a copy of this License along with the Program.
|
original licensor to copy, distribute or modify the Program subject to
|
||||||
|
these terms and conditions. You may not impose any further
|
||||||
You may charge any price or no price for each copy that you convey,
|
restrictions on the recipients' exercise of the rights granted herein.
|
||||||
and you may offer support or warranty protection for a fee.
|
You are not responsible for enforcing compliance by third parties to
|
||||||
|
|
||||||
5. Conveying Modified Source Versions.
|
|
||||||
|
|
||||||
You may convey a work based on the Program, or the modifications to
|
|
||||||
produce it from the Program, in the form of source code under the
|
|
||||||
terms of section 4, provided that you also meet all of these conditions:
|
|
||||||
|
|
||||||
a) The work must carry prominent notices stating that you modified
|
|
||||||
it, and giving a relevant date.
|
|
||||||
|
|
||||||
b) The work must carry prominent notices stating that it is
|
|
||||||
released under this License and any conditions added under section
|
|
||||||
7. This requirement modifies the requirement in section 4 to
|
|
||||||
"keep intact all notices".
|
|
||||||
|
|
||||||
c) You must license the entire work, as a whole, under this
|
|
||||||
License to anyone who comes into possession of a copy. This
|
|
||||||
License will therefore apply, along with any applicable section 7
|
|
||||||
additional terms, to the whole of the work, and all its parts,
|
|
||||||
regardless of how they are packaged. This License gives no
|
|
||||||
permission to license the work in any other way, but it does not
|
|
||||||
invalidate such permission if you have separately received it.
|
|
||||||
|
|
||||||
d) If the work has interactive user interfaces, each must display
|
|
||||||
Appropriate Legal Notices; however, if the Program has interactive
|
|
||||||
interfaces that do not display Appropriate Legal Notices, your
|
|
||||||
work need not make them do so.
|
|
||||||
|
|
||||||
A compilation of a covered work with other separate and independent
|
|
||||||
works, which are not by their nature extensions of the covered work,
|
|
||||||
and which are not combined with it such as to form a larger program,
|
|
||||||
in or on a volume of a storage or distribution medium, is called an
|
|
||||||
"aggregate" if the compilation and its resulting copyright are not
|
|
||||||
used to limit the access or legal rights of the compilation's users
|
|
||||||
beyond what the individual works permit. Inclusion of a covered work
|
|
||||||
in an aggregate does not cause this License to apply to the other
|
|
||||||
parts of the aggregate.
|
|
||||||
|
|
||||||
6. Conveying Non-Source Forms.
|
|
||||||
|
|
||||||
You may convey a covered work in object code form under the terms
|
|
||||||
of sections 4 and 5, provided that you also convey the
|
|
||||||
machine-readable Corresponding Source under the terms of this License,
|
|
||||||
in one of these ways:
|
|
||||||
|
|
||||||
a) Convey the object code in, or embodied in, a physical product
|
|
||||||
(including a physical distribution medium), accompanied by the
|
|
||||||
Corresponding Source fixed on a durable physical medium
|
|
||||||
customarily used for software interchange.
|
|
||||||
|
|
||||||
b) Convey the object code in, or embodied in, a physical product
|
|
||||||
(including a physical distribution medium), accompanied by a
|
|
||||||
written offer, valid for at least three years and valid for as
|
|
||||||
long as you offer spare parts or customer support for that product
|
|
||||||
model, to give anyone who possesses the object code either (1) a
|
|
||||||
copy of the Corresponding Source for all the software in the
|
|
||||||
product that is covered by this License, on a durable physical
|
|
||||||
medium customarily used for software interchange, for a price no
|
|
||||||
more than your reasonable cost of physically performing this
|
|
||||||
conveying of source, or (2) access to copy the
|
|
||||||
Corresponding Source from a network server at no charge.
|
|
||||||
|
|
||||||
c) Convey individual copies of the object code with a copy of the
|
|
||||||
written offer to provide the Corresponding Source. This
|
|
||||||
alternative is allowed only occasionally and noncommercially, and
|
|
||||||
only if you received the object code with such an offer, in accord
|
|
||||||
with subsection 6b.
|
|
||||||
|
|
||||||
d) Convey the object code by offering access from a designated
|
|
||||||
place (gratis or for a charge), and offer equivalent access to the
|
|
||||||
Corresponding Source in the same way through the same place at no
|
|
||||||
further charge. You need not require recipients to copy the
|
|
||||||
Corresponding Source along with the object code. If the place to
|
|
||||||
copy the object code is a network server, the Corresponding Source
|
|
||||||
may be on a different server (operated by you or a third party)
|
|
||||||
that supports equivalent copying facilities, provided you maintain
|
|
||||||
clear directions next to the object code saying where to find the
|
|
||||||
Corresponding Source. Regardless of what server hosts the
|
|
||||||
Corresponding Source, you remain obligated to ensure that it is
|
|
||||||
available for as long as needed to satisfy these requirements.
|
|
||||||
|
|
||||||
e) Convey the object code using peer-to-peer transmission, provided
|
|
||||||
you inform other peers where the object code and Corresponding
|
|
||||||
Source of the work are being offered to the general public at no
|
|
||||||
charge under subsection 6d.
|
|
||||||
|
|
||||||
A separable portion of the object code, whose source code is excluded
|
|
||||||
from the Corresponding Source as a System Library, need not be
|
|
||||||
included in conveying the object code work.
|
|
||||||
|
|
||||||
A "User Product" is either (1) a "consumer product", which means any
|
|
||||||
tangible personal property which is normally used for personal, family,
|
|
||||||
or household purposes, or (2) anything designed or sold for incorporation
|
|
||||||
into a dwelling. In determining whether a product is a consumer product,
|
|
||||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
|
||||||
product received by a particular user, "normally used" refers to a
|
|
||||||
typical or common use of that class of product, regardless of the status
|
|
||||||
of the particular user or of the way in which the particular user
|
|
||||||
actually uses, or expects or is expected to use, the product. A product
|
|
||||||
is a consumer product regardless of whether the product has substantial
|
|
||||||
commercial, industrial or non-consumer uses, unless such uses represent
|
|
||||||
the only significant mode of use of the product.
|
|
||||||
|
|
||||||
"Installation Information" for a User Product means any methods,
|
|
||||||
procedures, authorization keys, or other information required to install
|
|
||||||
and execute modified versions of a covered work in that User Product from
|
|
||||||
a modified version of its Corresponding Source. The information must
|
|
||||||
suffice to ensure that the continued functioning of the modified object
|
|
||||||
code is in no case prevented or interfered with solely because
|
|
||||||
modification has been made.
|
|
||||||
|
|
||||||
If you convey an object code work under this section in, or with, or
|
|
||||||
specifically for use in, a User Product, and the conveying occurs as
|
|
||||||
part of a transaction in which the right of possession and use of the
|
|
||||||
User Product is transferred to the recipient in perpetuity or for a
|
|
||||||
fixed term (regardless of how the transaction is characterized), the
|
|
||||||
Corresponding Source conveyed under this section must be accompanied
|
|
||||||
by the Installation Information. But this requirement does not apply
|
|
||||||
if neither you nor any third party retains the ability to install
|
|
||||||
modified object code on the User Product (for example, the work has
|
|
||||||
been installed in ROM).
|
|
||||||
|
|
||||||
The requirement to provide Installation Information does not include a
|
|
||||||
requirement to continue to provide support service, warranty, or updates
|
|
||||||
for a work that has been modified or installed by the recipient, or for
|
|
||||||
the User Product in which it has been modified or installed. Access to a
|
|
||||||
network may be denied when the modification itself materially and
|
|
||||||
adversely affects the operation of the network or violates the rules and
|
|
||||||
protocols for communication across the network.
|
|
||||||
|
|
||||||
Corresponding Source conveyed, and Installation Information provided,
|
|
||||||
in accord with this section must be in a format that is publicly
|
|
||||||
documented (and with an implementation available to the public in
|
|
||||||
source code form), and must require no special password or key for
|
|
||||||
unpacking, reading or copying.
|
|
||||||
|
|
||||||
7. Additional Terms.
|
|
||||||
|
|
||||||
"Additional permissions" are terms that supplement the terms of this
|
|
||||||
License by making exceptions from one or more of its conditions.
|
|
||||||
Additional permissions that are applicable to the entire Program shall
|
|
||||||
be treated as though they were included in this License, to the extent
|
|
||||||
that they are valid under applicable law. If additional permissions
|
|
||||||
apply only to part of the Program, that part may be used separately
|
|
||||||
under those permissions, but the entire Program remains governed by
|
|
||||||
this License without regard to the additional permissions.
|
|
||||||
|
|
||||||
When you convey a copy of a covered work, you may at your option
|
|
||||||
remove any additional permissions from that copy, or from any part of
|
|
||||||
it. (Additional permissions may be written to require their own
|
|
||||||
removal in certain cases when you modify the work.) You may place
|
|
||||||
additional permissions on material, added by you to a covered work,
|
|
||||||
for which you have or can give appropriate copyright permission.
|
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, for material you
|
|
||||||
add to a covered work, you may (if authorized by the copyright holders of
|
|
||||||
that material) supplement the terms of this License with terms:
|
|
||||||
|
|
||||||
a) Disclaiming warranty or limiting liability differently from the
|
|
||||||
terms of sections 15 and 16 of this License; or
|
|
||||||
|
|
||||||
b) Requiring preservation of specified reasonable legal notices or
|
|
||||||
author attributions in that material or in the Appropriate Legal
|
|
||||||
Notices displayed by works containing it; or
|
|
||||||
|
|
||||||
c) Prohibiting misrepresentation of the origin of that material, or
|
|
||||||
requiring that modified versions of such material be marked in
|
|
||||||
reasonable ways as different from the original version; or
|
|
||||||
|
|
||||||
d) Limiting the use for publicity purposes of names of licensors or
|
|
||||||
authors of the material; or
|
|
||||||
|
|
||||||
e) Declining to grant rights under trademark law for use of some
|
|
||||||
trade names, trademarks, or service marks; or
|
|
||||||
|
|
||||||
f) Requiring indemnification of licensors and authors of that
|
|
||||||
material by anyone who conveys the material (or modified versions of
|
|
||||||
it) with contractual assumptions of liability to the recipient, for
|
|
||||||
any liability that these contractual assumptions directly impose on
|
|
||||||
those licensors and authors.
|
|
||||||
|
|
||||||
All other non-permissive additional terms are considered "further
|
|
||||||
restrictions" within the meaning of section 10. If the Program as you
|
|
||||||
received it, or any part of it, contains a notice stating that it is
|
|
||||||
governed by this License along with a term that is a further
|
|
||||||
restriction, you may remove that term. If a license document contains
|
|
||||||
a further restriction but permits relicensing or conveying under this
|
|
||||||
License, you may add to a covered work material governed by the terms
|
|
||||||
of that license document, provided that the further restriction does
|
|
||||||
not survive such relicensing or conveying.
|
|
||||||
|
|
||||||
If you add terms to a covered work in accord with this section, you
|
|
||||||
must place, in the relevant source files, a statement of the
|
|
||||||
additional terms that apply to those files, or a notice indicating
|
|
||||||
where to find the applicable terms.
|
|
||||||
|
|
||||||
Additional terms, permissive or non-permissive, may be stated in the
|
|
||||||
form of a separately written license, or stated as exceptions;
|
|
||||||
the above requirements apply either way.
|
|
||||||
|
|
||||||
8. Termination.
|
|
||||||
|
|
||||||
You may not propagate or modify a covered work except as expressly
|
|
||||||
provided under this License. Any attempt otherwise to propagate or
|
|
||||||
modify it is void, and will automatically terminate your rights under
|
|
||||||
this License (including any patent licenses granted under the third
|
|
||||||
paragraph of section 11).
|
|
||||||
|
|
||||||
However, if you cease all violation of this License, then your
|
|
||||||
license from a particular copyright holder is reinstated (a)
|
|
||||||
provisionally, unless and until the copyright holder explicitly and
|
|
||||||
finally terminates your license, and (b) permanently, if the copyright
|
|
||||||
holder fails to notify you of the violation by some reasonable means
|
|
||||||
prior to 60 days after the cessation.
|
|
||||||
|
|
||||||
Moreover, your license from a particular copyright holder is
|
|
||||||
reinstated permanently if the copyright holder notifies you of the
|
|
||||||
violation by some reasonable means, this is the first time you have
|
|
||||||
received notice of violation of this License (for any work) from that
|
|
||||||
copyright holder, and you cure the violation prior to 30 days after
|
|
||||||
your receipt of the notice.
|
|
||||||
|
|
||||||
Termination of your rights under this section does not terminate the
|
|
||||||
licenses of parties who have received copies or rights from you under
|
|
||||||
this License. If your rights have been terminated and not permanently
|
|
||||||
reinstated, you do not qualify to receive new licenses for the same
|
|
||||||
material under section 10.
|
|
||||||
|
|
||||||
9. Acceptance Not Required for Having Copies.
|
|
||||||
|
|
||||||
You are not required to accept this License in order to receive or
|
|
||||||
run a copy of the Program. Ancillary propagation of a covered work
|
|
||||||
occurring solely as a consequence of using peer-to-peer transmission
|
|
||||||
to receive a copy likewise does not require acceptance. However,
|
|
||||||
nothing other than this License grants you permission to propagate or
|
|
||||||
modify any covered work. These actions infringe copyright if you do
|
|
||||||
not accept this License. Therefore, by modifying or propagating a
|
|
||||||
covered work, you indicate your acceptance of this License to do so.
|
|
||||||
|
|
||||||
10. Automatic Licensing of Downstream Recipients.
|
|
||||||
|
|
||||||
Each time you convey a covered work, the recipient automatically
|
|
||||||
receives a license from the original licensors, to run, modify and
|
|
||||||
propagate that work, subject to this License. You are not responsible
|
|
||||||
for enforcing compliance by third parties with this License.
|
|
||||||
|
|
||||||
An "entity transaction" is a transaction transferring control of an
|
|
||||||
organization, or substantially all assets of one, or subdividing an
|
|
||||||
organization, or merging organizations. If propagation of a covered
|
|
||||||
work results from an entity transaction, each party to that
|
|
||||||
transaction who receives a copy of the work also receives whatever
|
|
||||||
licenses to the work the party's predecessor in interest had or could
|
|
||||||
give under the previous paragraph, plus a right to possession of the
|
|
||||||
Corresponding Source of the work from the predecessor in interest, if
|
|
||||||
the predecessor has it or can get it with reasonable efforts.
|
|
||||||
|
|
||||||
You may not impose any further restrictions on the exercise of the
|
|
||||||
rights granted or affirmed under this License. For example, you may
|
|
||||||
not impose a license fee, royalty, or other charge for exercise of
|
|
||||||
rights granted under this License, and you may not initiate litigation
|
|
||||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
|
||||||
any patent claim is infringed by making, using, selling, offering for
|
|
||||||
sale, or importing the Program or any portion of it.
|
|
||||||
|
|
||||||
11. Patents.
|
|
||||||
|
|
||||||
A "contributor" is a copyright holder who authorizes use under this
|
|
||||||
License of the Program or a work on which the Program is based. The
|
|
||||||
work thus licensed is called the contributor's "contributor version".
|
|
||||||
|
|
||||||
A contributor's "essential patent claims" are all patent claims
|
|
||||||
owned or controlled by the contributor, whether already acquired or
|
|
||||||
hereafter acquired, that would be infringed by some manner, permitted
|
|
||||||
by this License, of making, using, or selling its contributor version,
|
|
||||||
but do not include claims that would be infringed only as a
|
|
||||||
consequence of further modification of the contributor version. For
|
|
||||||
purposes of this definition, "control" includes the right to grant
|
|
||||||
patent sublicenses in a manner consistent with the requirements of
|
|
||||||
this License.
|
this License.
|
||||||
|
|
||||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
7. If, as a consequence of a court judgment or allegation of patent
|
||||||
patent license under the contributor's essential patent claims, to
|
infringement or for any other reason (not limited to patent issues),
|
||||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
conditions are imposed on you (whether by court order, agreement or
|
||||||
propagate the contents of its contributor version.
|
|
||||||
|
|
||||||
In the following three paragraphs, a "patent license" is any express
|
|
||||||
agreement or commitment, however denominated, not to enforce a patent
|
|
||||||
(such as an express permission to practice a patent or covenant not to
|
|
||||||
sue for patent infringement). To "grant" such a patent license to a
|
|
||||||
party means to make such an agreement or commitment not to enforce a
|
|
||||||
patent against the party.
|
|
||||||
|
|
||||||
If you convey a covered work, knowingly relying on a patent license,
|
|
||||||
and the Corresponding Source of the work is not available for anyone
|
|
||||||
to copy, free of charge and under the terms of this License, through a
|
|
||||||
publicly available network server or other readily accessible means,
|
|
||||||
then you must either (1) cause the Corresponding Source to be so
|
|
||||||
available, or (2) arrange to deprive yourself of the benefit of the
|
|
||||||
patent license for this particular work, or (3) arrange, in a manner
|
|
||||||
consistent with the requirements of this License, to extend the patent
|
|
||||||
license to downstream recipients. "Knowingly relying" means you have
|
|
||||||
actual knowledge that, but for the patent license, your conveying the
|
|
||||||
covered work in a country, or your recipient's use of the covered work
|
|
||||||
in a country, would infringe one or more identifiable patents in that
|
|
||||||
country that you have reason to believe are valid.
|
|
||||||
|
|
||||||
If, pursuant to or in connection with a single transaction or
|
|
||||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
|
||||||
covered work, and grant a patent license to some of the parties
|
|
||||||
receiving the covered work authorizing them to use, propagate, modify
|
|
||||||
or convey a specific copy of the covered work, then the patent license
|
|
||||||
you grant is automatically extended to all recipients of the covered
|
|
||||||
work and works based on it.
|
|
||||||
|
|
||||||
A patent license is "discriminatory" if it does not include within
|
|
||||||
the scope of its coverage, prohibits the exercise of, or is
|
|
||||||
conditioned on the non-exercise of one or more of the rights that are
|
|
||||||
specifically granted under this License. You may not convey a covered
|
|
||||||
work if you are a party to an arrangement with a third party that is
|
|
||||||
in the business of distributing software, under which you make payment
|
|
||||||
to the third party based on the extent of your activity of conveying
|
|
||||||
the work, and under which the third party grants, to any of the
|
|
||||||
parties who would receive the covered work from you, a discriminatory
|
|
||||||
patent license (a) in connection with copies of the covered work
|
|
||||||
conveyed by you (or copies made from those copies), or (b) primarily
|
|
||||||
for and in connection with specific products or compilations that
|
|
||||||
contain the covered work, unless you entered into that arrangement,
|
|
||||||
or that patent license was granted, prior to 28 March 2007.
|
|
||||||
|
|
||||||
Nothing in this License shall be construed as excluding or limiting
|
|
||||||
any implied license or other defenses to infringement that may
|
|
||||||
otherwise be available to you under applicable patent law.
|
|
||||||
|
|
||||||
12. No Surrender of Others' Freedom.
|
|
||||||
|
|
||||||
If conditions are imposed on you (whether by court order, agreement or
|
|
||||||
otherwise) that contradict the conditions of this License, they do not
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
excuse you from the conditions of this License. If you cannot convey a
|
excuse you from the conditions of this License. If you cannot
|
||||||
covered work so as to satisfy simultaneously your obligations under this
|
distribute so as to satisfy simultaneously your obligations under this
|
||||||
License and any other pertinent obligations, then as a consequence you may
|
License and any other pertinent obligations, then as a consequence you
|
||||||
not convey it at all. For example, if you agree to terms that obligate you
|
may not distribute the Program at all. For example, if a patent
|
||||||
to collect a royalty for further conveying from those to whom you convey
|
license would not permit royalty-free redistribution of the Program by
|
||||||
the Program, the only way you could satisfy both those terms and this
|
all those who receive copies directly or indirectly through you, then
|
||||||
License would be to refrain entirely from conveying the Program.
|
the only way you could satisfy both it and this License would be to
|
||||||
|
refrain entirely from distribution of the Program.
|
||||||
|
|
||||||
13. Use with the GNU Affero General Public License.
|
If any portion of this section is held invalid or unenforceable under
|
||||||
|
any particular circumstance, the balance of the section is intended to
|
||||||
|
apply and the section as a whole is intended to apply in other
|
||||||
|
circumstances.
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, you have
|
It is not the purpose of this section to induce you to infringe any
|
||||||
permission to link or combine any covered work with a work licensed
|
patents or other property right claims or to contest validity of any
|
||||||
under version 3 of the GNU Affero General Public License into a single
|
such claims; this section has the sole purpose of protecting the
|
||||||
combined work, and to convey the resulting work. The terms of this
|
integrity of the free software distribution system, which is
|
||||||
License will continue to apply to the part which is the covered work,
|
implemented by public license practices. Many people have made
|
||||||
but the special requirements of the GNU Affero General Public License,
|
generous contributions to the wide range of software distributed
|
||||||
section 13, concerning interaction through a network will apply to the
|
through that system in reliance on consistent application of that
|
||||||
combination as such.
|
system; it is up to the author/donor to decide if he or she is willing
|
||||||
|
to distribute software through any other system and a licensee cannot
|
||||||
|
impose that choice.
|
||||||
|
|
||||||
14. Revised Versions of this License.
|
This section is intended to make thoroughly clear what is believed to
|
||||||
|
be a consequence of the rest of this License.
|
||||||
|
|
||||||
The Free Software Foundation may publish revised and/or new versions of
|
8. If the distribution and/or use of the Program is restricted in
|
||||||
the GNU General Public License from time to time. Such new versions will
|
certain countries either by patents or by copyrighted interfaces, the
|
||||||
|
original copyright holder who places the Program under this License
|
||||||
|
may add an explicit geographical distribution limitation excluding
|
||||||
|
those countries, so that distribution is permitted only in or among
|
||||||
|
countries not thus excluded. In such case, this License incorporates
|
||||||
|
the limitation as if written in the body of this License.
|
||||||
|
|
||||||
|
9. The Free Software Foundation may publish revised and/or new versions
|
||||||
|
of the General Public License from time to time. Such new versions will
|
||||||
be similar in spirit to the present version, but may differ in detail to
|
be similar in spirit to the present version, but may differ in detail to
|
||||||
address new problems or concerns.
|
address new problems or concerns.
|
||||||
|
|
||||||
Each version is given a distinguishing version number. If the
|
Each version is given a distinguishing version number. If the Program
|
||||||
Program specifies that a certain numbered version of the GNU General
|
specifies a version number of this License which applies to it and "any
|
||||||
Public License "or any later version" applies to it, you have the
|
later version", you have the option of following the terms and conditions
|
||||||
option of following the terms and conditions either of that numbered
|
either of that version or of any later version published by the Free
|
||||||
version or of any later version published by the Free Software
|
Software Foundation. If the Program does not specify a version number of
|
||||||
Foundation. If the Program does not specify a version number of the
|
this License, you may choose any version ever published by the Free Software
|
||||||
GNU General Public License, you may choose any version ever published
|
Foundation.
|
||||||
by the Free Software Foundation.
|
|
||||||
|
|
||||||
If the Program specifies that a proxy can decide which future
|
10. If you wish to incorporate parts of the Program into other free
|
||||||
versions of the GNU General Public License can be used, that proxy's
|
programs whose distribution conditions are different, write to the author
|
||||||
public statement of acceptance of a version permanently authorizes you
|
to ask for permission. For software which is copyrighted by the Free
|
||||||
to choose that version for the Program.
|
Software Foundation, write to the Free Software Foundation; we sometimes
|
||||||
|
make exceptions for this. Our decision will be guided by the two goals
|
||||||
|
of preserving the free status of all derivatives of our free software and
|
||||||
|
of promoting the sharing and reuse of software generally.
|
||||||
|
|
||||||
Later license versions may give you additional or different
|
NO WARRANTY
|
||||||
permissions. However, no additional obligations are imposed on any
|
|
||||||
author or copyright holder as a result of your choosing to follow a
|
|
||||||
later version.
|
|
||||||
|
|
||||||
15. Disclaimer of Warranty.
|
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
|
||||||
|
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
|
||||||
|
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
|
||||||
|
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
|
||||||
|
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
|
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
|
||||||
|
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
|
||||||
|
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
|
||||||
|
REPAIR OR CORRECTION.
|
||||||
|
|
||||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
||||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
||||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
|
||||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
|
||||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
|
||||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
|
||||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
|
||||||
|
POSSIBILITY OF SUCH DAMAGES.
|
||||||
16. Limitation of Liability.
|
|
||||||
|
|
||||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
|
||||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
|
||||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
|
||||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
|
||||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
|
||||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
|
||||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
|
||||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
|
||||||
SUCH DAMAGES.
|
|
||||||
|
|
||||||
17. Interpretation of Sections 15 and 16.
|
|
||||||
|
|
||||||
If the disclaimer of warranty and limitation of liability provided
|
|
||||||
above cannot be given local legal effect according to their terms,
|
|
||||||
reviewing courts shall apply local law that most closely approximates
|
|
||||||
an absolute waiver of all civil liability in connection with the
|
|
||||||
Program, unless a warranty or assumption of liability accompanies a
|
|
||||||
copy of the Program in return for a fee.
|
|
||||||
|
|
||||||
END OF TERMS AND CONDITIONS
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
@@ -628,15 +287,15 @@ free software which everyone can redistribute and change under these terms.
|
|||||||
|
|
||||||
To do so, attach the following notices to the program. It is safest
|
To do so, attach the following notices to the program. It is safest
|
||||||
to attach them to the start of each source file to most effectively
|
to attach them to the start of each source file to most effectively
|
||||||
state the exclusion of warranty; and each file should have at least
|
convey the exclusion of warranty; and each file should have at least
|
||||||
the "copyright" line and a pointer to where the full notice is found.
|
the "copyright" line and a pointer to where the full notice is found.
|
||||||
|
|
||||||
<one line to give the program's name and a brief idea of what it does.>
|
<one line to give the program's name and a brief idea of what it does.>
|
||||||
Copyright (C) <year> <name of author>
|
Copyright (C) <year> <name of author>
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software; you can redistribute it and/or modify
|
||||||
it under the terms of the GNU General Public License as published by
|
it under the terms of the GNU General Public License as published by
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
the Free Software Foundation; either version 2 of the License, or
|
||||||
(at your option) any later version.
|
(at your option) any later version.
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
@@ -644,31 +303,37 @@ the "copyright" line and a pointer to where the full notice is found.
|
|||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
GNU General Public License for more details.
|
GNU General Public License for more details.
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
You should have received a copy of the GNU General Public License along
|
||||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
|
||||||
Also add information on how to contact you by electronic and paper mail.
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
If the program does terminal interaction, make it output a short
|
If the program is interactive, make it output a short notice like this
|
||||||
notice like this when it starts in an interactive mode:
|
when it starts in an interactive mode:
|
||||||
|
|
||||||
<program> Copyright (C) <year> <name of author>
|
Gnomovision version 69, Copyright (C) year name of author
|
||||||
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||||
This is free software, and you are welcome to redistribute it
|
This is free software, and you are welcome to redistribute it
|
||||||
under certain conditions; type `show c' for details.
|
under certain conditions; type `show c' for details.
|
||||||
|
|
||||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||||
parts of the General Public License. Of course, your program's commands
|
parts of the General Public License. Of course, the commands you use may
|
||||||
might be different; for a GUI interface, you would use an "about box".
|
be called something other than `show w' and `show c'; they could even be
|
||||||
|
mouse-clicks or menu items--whatever suits your program.
|
||||||
|
|
||||||
You should also get your employer (if you work as a programmer) or school,
|
You should also get your employer (if you work as a programmer) or your
|
||||||
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
school, if any, to sign a "copyright disclaimer" for the program, if
|
||||||
For more information on this, and how to apply and follow the GNU GPL, see
|
necessary. Here is a sample; alter the names:
|
||||||
<https://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
The GNU General Public License does not permit incorporating your program
|
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||||
into proprietary programs. If your program is a subroutine library, you
|
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||||
may consider it more useful to permit linking proprietary applications with
|
|
||||||
the library. If this is what you want to do, use the GNU Lesser General
|
<signature of Ty Coon>, 1 April 1989
|
||||||
Public License instead of this License. But first, please read
|
Ty Coon, President of Vice
|
||||||
<https://www.gnu.org/licenses/why-not-lgpl.html>.
|
|
||||||
|
This General Public License does not permit incorporating your program into
|
||||||
|
proprietary programs. If your program is a subroutine library, you may
|
||||||
|
consider it more useful to permit linking proprietary applications with the
|
||||||
|
library. If this is what you want to do, use the GNU Lesser General
|
||||||
|
Public License instead of this License.
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ recursive-include data *
|
|||||||
recursive-include src/revpicommander *.py *.qm
|
recursive-include src/revpicommander *.py *.qm
|
||||||
recursive-include ui_dev *
|
recursive-include ui_dev *
|
||||||
include LICENSE.txt
|
include LICENSE.txt
|
||||||
include make_installer_win.bat
|
include make.bat
|
||||||
include Makefile
|
include Makefile
|
||||||
include MANIFEST.in
|
include MANIFEST.in
|
||||||
include README.md
|
include README.md
|
||||||
|
|||||||
111
Makefile
111
Makefile
@@ -4,80 +4,125 @@ MAKEFLAGS = --no-print-directory --no-builtin-rules
|
|||||||
|
|
||||||
# Variables
|
# Variables
|
||||||
PACKAGE = revpicommander
|
PACKAGE = revpicommander
|
||||||
|
APP_NAME = RevPi\ Commander
|
||||||
|
APP_IDENT = org.revpimodio.revpicommander
|
||||||
|
APPLE_SIG = "Developer ID Application: Sven Sager (U3N5843D9K)"
|
||||||
|
|
||||||
# If virtualenv exists, use it. If not, use PATH to find
|
# Set path to create the virtual environment with package name
|
||||||
SYSTEM_PYTHON = $(or $(shell which python3), $(shell which python))
|
ifdef PYTHON3_VENV
|
||||||
PYTHON = $(or $(wildcard venv/bin/python), $(SYSTEM_PYTHON))
|
VENV_PATH = $(PYTHON3_VENV)/$(PACKAGE)
|
||||||
SYSTEM_PYUIC5 = $(shell which pyuic5)
|
else
|
||||||
PYUIC5 = $(or $(wildcard venv/bin/pyuic5), $(SYSTEM_PYUIC5))
|
VENV_PATH = venv
|
||||||
SYSTEM_PYRCC5 = $(shell which pyrcc5)
|
endif
|
||||||
PYRCC5 = $(or $(wildcard venv/bin/pyrcc5), $(SYSTEM_PYRCC5))
|
|
||||||
SYSTEM_PYLUP5 = $(shell which pylupdate5)
|
|
||||||
PYLUP5 = $(or $(wildcard venv/bin/pylupdate5), $(SYSTEM_PYLUP5))
|
|
||||||
|
|
||||||
all: build_ui build_rc build
|
# If virtualenv exists, use it. If not, use PATH to find commands
|
||||||
|
SYSTEM_PYTHON = python3
|
||||||
|
PYTHON = $(or $(wildcard $(VENV_PATH)/bin/python), $(SYSTEM_PYTHON))
|
||||||
|
|
||||||
|
APP_VERSION = $(shell $(PYTHON) src/$(PACKAGE) --version)
|
||||||
|
|
||||||
|
all: build_ui build_rc test build
|
||||||
|
|
||||||
.PHONY: all
|
.PHONY: all
|
||||||
|
|
||||||
|
## Environment
|
||||||
|
venv-info:
|
||||||
|
echo Using path: "$(VENV_PATH)"
|
||||||
|
exit 0
|
||||||
|
|
||||||
|
venv:
|
||||||
|
$(SYSTEM_PYTHON) -m venv --system-site-packages "$(VENV_PATH)"
|
||||||
|
source $(VENV_PATH)/bin/activate && \
|
||||||
|
python3 -m pip install --upgrade pip && \
|
||||||
|
python3 -m pip install -r requirements.txt
|
||||||
|
exit 0
|
||||||
|
|
||||||
|
.PHONY: venv-info venv
|
||||||
|
|
||||||
## Compile Qt UI files to python code
|
## Compile Qt UI files to python code
|
||||||
build_ui:
|
build_ui:
|
||||||
cd ui_dev && for ui_file in *.ui; do \
|
cd ui_dev && for ui_file in *.ui; do \
|
||||||
file_name=$${ui_file%.ui}; \
|
file_name=$${ui_file%.ui}; \
|
||||||
$(PYUIC5) $${ui_file} -o ../src/$(PACKAGE)/ui/$${file_name}_ui.py -x --from-imports; \
|
$(PYTHON) -m PyQt5.uic.pyuic $${ui_file} -o ../src/$(PACKAGE)/ui/$${file_name}_ui.py -x --from-imports; \
|
||||||
echo $${file_name}; \
|
echo $${file_name}; \
|
||||||
done
|
done
|
||||||
|
|
||||||
build_rc:
|
build_rc:
|
||||||
cd ui_dev && for rc_file in *.qrc; do \
|
cd ui_dev && for rc_file in *.qrc; do \
|
||||||
file_name=$${rc_file%.qrc}; \
|
file_name=$${rc_file%.qrc}; \
|
||||||
$(PYRCC5) $${rc_file} -o ../src/$(PACKAGE)/ui/$${file_name}_rc.py; \
|
$(PYTHON) -m PyQt5.pyrcc_main $${rc_file} -o ../src/$(PACKAGE)/ui/$${file_name}_rc.py; \
|
||||||
echo $${file_name}; \
|
echo $${file_name}; \
|
||||||
done
|
done
|
||||||
|
|
||||||
update_translation:
|
update_translation:
|
||||||
$(PYLUP5) translate.pro
|
$(PYTHON) -m PyQt5.pylupdate_main translate.pro
|
||||||
|
|
||||||
.PHONY: build_ui build_rc update_translation
|
.PHONY: build_ui build_rc update_translation
|
||||||
|
|
||||||
## Environment
|
## Build steps
|
||||||
venv:
|
|
||||||
rm -rf venv
|
|
||||||
$(SYSTEM_PYTHON) -m venv venv
|
|
||||||
|
|
||||||
deps:
|
|
||||||
$(PYTHON) -m pip install --upgrade pip -r requirements.txt
|
|
||||||
|
|
||||||
.PHONY: venv deps
|
|
||||||
|
|
||||||
## Build, install
|
|
||||||
build: build_ui build_rc
|
build: build_ui build_rc
|
||||||
$(PYTHON) -m setup sdist
|
$(PYTHON) -m setup sdist
|
||||||
$(PYTHON) -m setup bdist_wheel
|
$(PYTHON) -m setup bdist_wheel
|
||||||
|
|
||||||
install:
|
install: test build
|
||||||
$(PYTHON) -m pip install dist/$(PACKAGE)-*.whl
|
$(PYTHON) -m pip install dist/$(PACKAGE)-*.whl
|
||||||
|
|
||||||
.PHONY: build install
|
uninstall:
|
||||||
|
$(PYTHON) -m pip uninstall --yes $(PACKAGE)
|
||||||
|
|
||||||
|
.PHONY: test build install uninstall
|
||||||
|
|
||||||
## PyInstaller
|
## PyInstaller
|
||||||
installer_mac: build
|
installer_mac: build_ui build_rc
|
||||||
$(PYTHON) -m PyInstaller -n "RevPi Commander" \
|
$(PYTHON) -m PyInstaller -n $(APP_NAME) \
|
||||||
--add-data="src/$(PACKAGE)/locale:./revpicommander/locale" \
|
--add-data="src/$(PACKAGE)/locale:./$(PACKAGE)/locale" \
|
||||||
--add-data="data/$(PACKAGE).icns:." \
|
--add-data="data/$(PACKAGE).icns:." \
|
||||||
--icon=data/$(PACKAGE).icns \
|
--icon=data/$(PACKAGE).icns \
|
||||||
--noconfirm \
|
--noconfirm \
|
||||||
--clean \
|
--clean \
|
||||||
--onedir \
|
--onedir \
|
||||||
--windowed \
|
--windowed \
|
||||||
|
--osx-bundle-identifier $APP_IDENT \
|
||||||
|
--codesign-identity $(APPLE_SIG) \
|
||||||
src/$(PACKAGE)/__main__.py
|
src/$(PACKAGE)/__main__.py
|
||||||
|
|
||||||
installer_win: all
|
installer_mac_dmg: installer_mac
|
||||||
make_installer_win.bat
|
mkdir dist/dmg
|
||||||
|
mv dist/$(APP_NAME).app dist/dmg
|
||||||
|
create-dmg \
|
||||||
|
--volname $(APP_NAME) \
|
||||||
|
--background data/dmg_background.png \
|
||||||
|
--window-pos 200 120 \
|
||||||
|
--window-size 480 300 \
|
||||||
|
--icon-size 64 \
|
||||||
|
--icon $(APP_NAME).app 64 64 \
|
||||||
|
--hide-extension $(APP_NAME).app \
|
||||||
|
--app-drop-link 288 64 \
|
||||||
|
--add-file LICENSE.txt LICENSE.txt 192 180 \
|
||||||
|
--codesign $(APPLE_SIG) \
|
||||||
|
--notarize AC_PASSWORD \
|
||||||
|
dist/$(APP_NAME)\ $(APP_VERSION).dmg \
|
||||||
|
dist/dmg
|
||||||
|
|
||||||
.PHONY: installer_mac installer_win
|
installer_linux: build_ui build_rc
|
||||||
|
$(PYTHON) -m PyInstaller -n $(APP_NAME) \
|
||||||
|
--add-data="src/$(PACKAGE)/locale:./$(PACKAGE)/locale" \
|
||||||
|
--add-data="data/$(PACKAGE).ico:." \
|
||||||
|
--add-data="data/$(PACKAGE).png:." \
|
||||||
|
--icon=data/$(PACKAGE).ico \
|
||||||
|
--noconfirm \
|
||||||
|
--clean \
|
||||||
|
--onedir \
|
||||||
|
--windowed \
|
||||||
|
src/$(PACKAGE)/__main__.py
|
||||||
|
|
||||||
|
.PHONY: installer_mac installer_mac_dmg installer_linux
|
||||||
|
|
||||||
## Clean
|
## Clean
|
||||||
clean:
|
clean:
|
||||||
rm -rf build dist src/*.egg-info *.spec
|
rm -rf build dist src/*.egg-info *.spec
|
||||||
|
|
||||||
.PHONY: clean
|
distclean: clean
|
||||||
|
rm -rf $(VENV_PATH)
|
||||||
|
|
||||||
|
.PHONY: clean distclean
|
||||||
|
|||||||
BIN
data/dmg_background.png
Normal file
BIN
data/dmg_background.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 172 KiB |
@@ -1,3 +1,7 @@
|
|||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
|
# This script will create the entry point 'revpicommander', which is defined
|
||||||
|
# in the setup.py script. If you want to install this python module as an
|
||||||
|
# application and not in the python standard library directory, you will need
|
||||||
|
# to use this script for /usr/bin.
|
||||||
|
|
||||||
exec "/usr/share/revpicommander/revpicommander.py" "$@"
|
exec python3 /usr/share/revpicommander/revpicommander "$@"
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
[Desktop Entry]
|
[Desktop Entry]
|
||||||
Name=RevPi PLC Commander
|
Name=RevPi Commander
|
||||||
Comment=Controls the Python PLC program on your Revolution PI
|
Comment=Controls the Python PLC program on your Revolution PI
|
||||||
Comment[de]=Kontrolliert das Python PLC Programm auf dem Revolution PI
|
Comment[de]=Kontrolliert das Python PLC Programm auf dem Revolution PI
|
||||||
Exec=/usr/bin/revpicommander
|
Exec=/usr/bin/revpicommander
|
||||||
@@ -7,4 +7,3 @@ Icon=revpicommander
|
|||||||
Terminal=false
|
Terminal=false
|
||||||
Type=Application
|
Type=Application
|
||||||
Categories=Application;
|
Categories=Application;
|
||||||
#StartupNotify=true
|
|
||||||
|
|||||||
Binary file not shown.
Binary file not shown.
|
Before Width: | Height: | Size: 1.4 KiB After Width: | Height: | Size: 5.3 KiB |
98
debian/changelog
vendored
Normal file
98
debian/changelog
vendored
Normal file
@@ -0,0 +1,98 @@
|
|||||||
|
revpicommander (0.10.0-2) stable; urgency=medium
|
||||||
|
|
||||||
|
[ Sven Sager ]
|
||||||
|
* Renamed version of revpipycontrol written with Qt framework
|
||||||
|
* Replace widget search to widget directory (performance)
|
||||||
|
* Replace widget search to widget dict (performance)
|
||||||
|
* WIP: GUI and base function from revpidevelop
|
||||||
|
* Bugfix: PyInstaller script
|
||||||
|
* Save IP address, not hostname of avahisearch.py found RevPi
|
||||||
|
(Windows will not resolve)
|
||||||
|
* WIP: revpifiles.py has all functions for PyLoad 0.9.2
|
||||||
|
* revpifiles.py is ready with PyLoad 0.9.3 functions
|
||||||
|
* files.ui Design change, cleanup
|
||||||
|
* String fields (> 4 Bytes) can be switched to numbers
|
||||||
|
* Load and display logfiles block by block
|
||||||
|
* Add option for software watchdog and driver reset action
|
||||||
|
* Destroy changed IOs after piCtory changes
|
||||||
|
* UI Text check and translation-de
|
||||||
|
* Bugfixes - IP regex, Queue handling, file upload
|
||||||
|
* UI Text check and translation-de
|
||||||
|
* Bugfix: Load locales file
|
||||||
|
* Remove included revpimodio2 static link and add it as requires
|
||||||
|
* New release
|
||||||
|
* Get AVAHI IPv4 addresses with service_info
|
||||||
|
* Bugfix for translations strings and format function
|
||||||
|
* Add parameter to change connection timeout (default was and
|
||||||
|
is 5 seconds)
|
||||||
|
* Add a simulator dialog to manage simulator settings
|
||||||
|
* Bugfix for simulator to prevent GUI crash after first use
|
||||||
|
* Modified translations
|
||||||
|
* New release
|
||||||
|
* Add upload progress display in developer, bugfix on file download in
|
||||||
|
developer
|
||||||
|
* Fix error on decoding broken log files
|
||||||
|
* Fix crash during add new connection
|
||||||
|
* Reduce max block to load for log files on slow networks
|
||||||
|
* Improved RevPi search window
|
||||||
|
* New release
|
||||||
|
* Fix error in debugios.py with long byte values
|
||||||
|
* Bugfix: RevPi simulator could not be startet without existing
|
||||||
|
proc.img file
|
||||||
|
* Design things: enable HighDpi, dependency versions, icon text
|
||||||
|
* Add Windows files for pyinstaller and setup generation
|
||||||
|
* Fix qt parents of menu entries in connections
|
||||||
|
* Improved shortcuts to meet the needs of different operating systems
|
||||||
|
* Implement update function of ZeroConf ServiceListener
|
||||||
|
* Redesign main window with text boxes to copy host name or ip
|
||||||
|
* Add translation of status codes
|
||||||
|
* Bugfix: could not open debugcontrol, because of new layout names
|
||||||
|
* Add context menu with copy and piCtory actions to revpi search dialog
|
||||||
|
* Switch project so src layout
|
||||||
|
* Switch to setuptools for setup.py
|
||||||
|
* Add icons to commander main window for start, stop and so on
|
||||||
|
* Update Makefile and requirements.txt for build and clean ui files
|
||||||
|
* Add PyInstaller script for Windows
|
||||||
|
* QTranslator uses ui_languages to find translation
|
||||||
|
* Use cm.call_remote_function to fetch log files via XML-RPC
|
||||||
|
* Prepare SSH tunnel server and password GUI
|
||||||
|
* Configure and start ssh tunneled connections
|
||||||
|
* Choose connection type (ssh or xml-rpc) in avahisearch dialog
|
||||||
|
* Update Makefile rules
|
||||||
|
* Update translation file
|
||||||
|
* Update Windows setup script to match pyinstaller output
|
||||||
|
* Manage revolution pi saved settings in an own class
|
||||||
|
* Change most button pressed event to clicked and add global shortcuts
|
||||||
|
* Set the use of ssh tunnel to default value for new settings
|
||||||
|
* Save SSH credentials with keyring module
|
||||||
|
* Show just the hostname without .local in avahisearch.py list
|
||||||
|
* Bugfix for new settings management
|
||||||
|
* Clean up keyring when a connection is deleted
|
||||||
|
* Move version number from revpicommander.py to __init__.py
|
||||||
|
* New async connection management with user feedback
|
||||||
|
* Connection list with add folder button and removed naming bug
|
||||||
|
* Move version number from __init__.py to __main__.py
|
||||||
|
* Adjustments for creating installations on various operating systems
|
||||||
|
* Bugfix: Add missing packages to install_requires
|
||||||
|
* Add option to switch word order on debugios
|
||||||
|
* Add pyinstaller option for linux platform
|
||||||
|
* Bugfix type hinting, PyLoad version handling, piCtory browser
|
||||||
|
* Update translation and messages
|
||||||
|
* Change license from GPLv3 to GPLv2 after approval of all contributors
|
||||||
|
* Replaces Master-Slave with Client-Server
|
||||||
|
* Bugfix in super call of backgroundworker.py
|
||||||
|
* Ignore ssh known_hosts from home directory, use own
|
||||||
|
* build: Use system python interpreter in make file
|
||||||
|
* fix: Show program dialog, even if there are no files on the RevPi
|
||||||
|
* chore: Release 0.10.0rc1
|
||||||
|
* fix: PLC monitor crashes when a device has no inputs or outputs
|
||||||
|
* build: Add environment variable to set alternative venv path
|
||||||
|
* feat: SSH tunnel server extended to execute commands on remote host.
|
||||||
|
* build: Use python interpreter and modules for qt commands in Makefile
|
||||||
|
* feat: Try to start RevPiPyLoad via SSH on the Revolution Pi
|
||||||
|
* fix: Simulator always sets process image to NULL
|
||||||
|
* build: Fix Makefile targets to match GNU coding standards
|
||||||
|
* chore: Release 0.10.0
|
||||||
|
* packaging(deb): Start packaging branch
|
||||||
|
|
||||||
|
-- Sven Sager <s.sager@kunbus.com> Sat, 16 Sep 2023 14:10:12 +0200
|
||||||
32
debian/control
vendored
Normal file
32
debian/control
vendored
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
Source: revpicommander
|
||||||
|
Section: x11
|
||||||
|
Priority: optional
|
||||||
|
Maintainer: Sven Sager <akira@narux.de>
|
||||||
|
Rules-Requires-Root: no
|
||||||
|
Homepage: https://revpimodio.org/revpipyplc/
|
||||||
|
Vcs-Browser: https://github.com/narux/revpicommander
|
||||||
|
Vcs-Git: https://github.com/narux/revpicommander.git
|
||||||
|
Build-Depends:
|
||||||
|
debhelper-compat (= 12),
|
||||||
|
dh-python,
|
||||||
|
python3-all,
|
||||||
|
python3-keyring (>= 17.1.1),
|
||||||
|
python3-pyqt5,
|
||||||
|
python3-paramiko (>= 2.4.2),
|
||||||
|
python3-revpimodio2 (>= 2.5.6),
|
||||||
|
python3-zeroconf (>= 0.24.4),
|
||||||
|
python3-setuptools,
|
||||||
|
python3-wheel
|
||||||
|
Standards-Version: 4.3.0
|
||||||
|
|
||||||
|
Package: revpicommander
|
||||||
|
Architecture: all
|
||||||
|
Depends:
|
||||||
|
${python3:Depends},
|
||||||
|
${misc:Depends}
|
||||||
|
Description: GUI for Revolution Pi to upload programs and do IO-Checks
|
||||||
|
The RevPiCommander is a GUI tool to manage your revolution Pi over the
|
||||||
|
network. You can search for RevPis in your network, manage the settings
|
||||||
|
of RevPiPyLoad and do IO checks on your local machine. Developing your
|
||||||
|
control program is very easy with the developer, upload and debug it
|
||||||
|
over the network.
|
||||||
46
debian/copyright
vendored
Normal file
46
debian/copyright
vendored
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
Format: https://www.debian.org/doc/packaging-manuals/copyright-format/1.0/
|
||||||
|
Source: https://github.com/naruxde/revpicommander
|
||||||
|
Upstream-Name: revpicommander
|
||||||
|
Upstream-Contact: Sven Sager <akira@narux.de>
|
||||||
|
|
||||||
|
Files: *
|
||||||
|
Copyright: 2017-2023 Sven Sager
|
||||||
|
License: GPL-2+
|
||||||
|
|
||||||
|
Files: debian/*
|
||||||
|
Copyright: 2021-2023 KUNBUS GmbH
|
||||||
|
License: GPL-2+
|
||||||
|
|
||||||
|
License: GPL-2+
|
||||||
|
This package is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation; either version 2 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
.
|
||||||
|
This package is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
.
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program. If not, see <https://www.gnu.org/licenses/>
|
||||||
|
.
|
||||||
|
On Debian systems, the complete text of the GNU General
|
||||||
|
Public License version 2 can be found in "/usr/share/common-licenses/GPL-2".
|
||||||
|
|
||||||
|
License: GPL-3+
|
||||||
|
This program is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
.
|
||||||
|
This package is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
.
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
.
|
||||||
|
On Debian systems, the complete text of the GNU General
|
||||||
|
Public License version 3 can be found in "/usr/share/common-licenses/GPL-3".
|
||||||
4
debian/gbp.conf
vendored
Normal file
4
debian/gbp.conf
vendored
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
[DEFAULT]
|
||||||
|
upstream-branch = main
|
||||||
|
upstream-tag = %(version)s
|
||||||
|
debian-branch=pkg/debian
|
||||||
3
debian/install
vendored
Normal file
3
debian/install
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
data/revpicommander /usr/bin
|
||||||
|
data/revpicommander.desktop /usr/share/applications
|
||||||
|
data/revpicommander.png /usr/share/icons/hicolor/128x128/apps
|
||||||
1
debian/py3dist-overrides
vendored
Normal file
1
debian/py3dist-overrides
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
pyqt5 python3-pyqt5
|
||||||
9
debian/rules
vendored
Executable file
9
debian/rules
vendored
Executable file
@@ -0,0 +1,9 @@
|
|||||||
|
#!/usr/bin/make -f
|
||||||
|
|
||||||
|
export PYBUILD_NAME=revpicommander
|
||||||
|
|
||||||
|
# Install this module as an application an not into the python standard library path
|
||||||
|
export PYBUILD_INSTALL_ARGS=--install-lib=/usr/share/revpicommander/ --install-scripts=/usr/share/revpicommander/bin/
|
||||||
|
|
||||||
|
%:
|
||||||
|
dh $@ --with python3 --buildsystem=pybuild
|
||||||
1
debian/source/format
vendored
Normal file
1
debian/source/format
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
3.0 (quilt)
|
||||||
39
make.bat
Normal file
39
make.bat
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
@echo off
|
||||||
|
set PACKAGE=revpicommander
|
||||||
|
set APP_NAME=RevPi Commander
|
||||||
|
|
||||||
|
if "%1" == "venv" goto venv
|
||||||
|
if "%1" == "installer" goto installer
|
||||||
|
if "%1" == "clean" goto clean
|
||||||
|
|
||||||
|
echo Make script for "%APP_NAME%" on Windows
|
||||||
|
echo.
|
||||||
|
echo Need action:
|
||||||
|
echo venv Create / update your virtual environment for build process
|
||||||
|
echo installer Build this application with PyInstaller
|
||||||
|
echo clean Clean up your environment after build process
|
||||||
|
goto end
|
||||||
|
|
||||||
|
:venv
|
||||||
|
python -m venv venv
|
||||||
|
venv\\Scripts\\pip.exe install --upgrade -r requirements.txt
|
||||||
|
goto end
|
||||||
|
|
||||||
|
:installer
|
||||||
|
venv\\Scripts\\pyinstaller -n "%APP_NAME%" ^
|
||||||
|
--add-data="src\%PACKAGE%\locale;.\%PACKAGE%\locale" ^
|
||||||
|
--add-data="data\%PACKAGE%.ico;." ^
|
||||||
|
--icon=data\\%PACKAGE%.ico ^
|
||||||
|
--noconfirm ^
|
||||||
|
--clean ^
|
||||||
|
--onedir ^
|
||||||
|
--windowed ^
|
||||||
|
src\\%PACKAGE%\\__main__.py
|
||||||
|
goto end
|
||||||
|
|
||||||
|
:clean
|
||||||
|
rmdir /S /Q build dist
|
||||||
|
rmdir /S /Q src\%PACKAGE%.egg-info
|
||||||
|
del *.spec
|
||||||
|
|
||||||
|
:end
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
@echo off
|
|
||||||
set PACKAGE=revpicommander
|
|
||||||
|
|
||||||
rem python -m venv venv
|
|
||||||
rem venv\\Scripts\\activate.bat
|
|
||||||
rem pip install -r requirements.txt
|
|
||||||
|
|
||||||
pyinstaller -n "RevPi Commander" ^
|
|
||||||
--add-data="src\%PACKAGE%\locale;.\revpicommander\locale" ^
|
|
||||||
--add-data="data\%PACKAGE%.ico;." ^
|
|
||||||
--icon=data\\%PACKAGE%.ico ^
|
|
||||||
--noconfirm ^
|
|
||||||
--clean ^
|
|
||||||
--onedir ^
|
|
||||||
--windowed ^
|
|
||||||
src\\%PACKAGE%\\__main__.py
|
|
||||||
|
|
||||||
rem deactivate
|
|
||||||
@@ -1,8 +1,9 @@
|
|||||||
Pyinstaller
|
|
||||||
PyQt5>=5.14.1
|
|
||||||
revpimodio2>=2.5.6
|
|
||||||
zeroconf>=0.24.4
|
|
||||||
setuptools>=65.6.3
|
setuptools>=65.6.3
|
||||||
wheel
|
wheel
|
||||||
paramiko>=2.12.0
|
Pyinstaller
|
||||||
|
|
||||||
keyring>=23.13.1
|
keyring>=23.13.1
|
||||||
|
PyQt5>=5.14.1
|
||||||
|
paramiko>=2.12.0
|
||||||
|
revpimodio2>=2.5.6
|
||||||
|
zeroconf>=0.24.4
|
||||||
|
|||||||
3
setup.iss
Executable file → Normal file
3
setup.iss
Executable file → Normal file
@@ -2,12 +2,13 @@
|
|||||||
; SEE THE DOCUMENTATION FOR DETAILS ON CREATING INNO SETUP SCRIPT FILES!
|
; SEE THE DOCUMENTATION FOR DETAILS ON CREATING INNO SETUP SCRIPT FILES!
|
||||||
|
|
||||||
#define MyAppName "RevPi Commander"
|
#define MyAppName "RevPi Commander"
|
||||||
#define MyAppVersion "0.9.10rc2"
|
#define MyAppVersion "0.10.0"
|
||||||
#define MyAppPublisher "Sven Sager"
|
#define MyAppPublisher "Sven Sager"
|
||||||
#define MyAppURL "https://revpimodio.org/"
|
#define MyAppURL "https://revpimodio.org/"
|
||||||
#define MyAppICO "data\revpicommander.ico"
|
#define MyAppICO "data\revpicommander.ico"
|
||||||
|
|
||||||
[Setup]
|
[Setup]
|
||||||
|
SignTool=kSign
|
||||||
; NOTE: The value of AppId uniquely identifies this application. Do not use the same AppId value in installers for other applications.
|
; NOTE: The value of AppId uniquely identifies this application. Do not use the same AppId value in installers for other applications.
|
||||||
; (To generate a new GUID, click Tools | Generate GUID inside the IDE.)
|
; (To generate a new GUID, click Tools | Generate GUID inside the IDE.)
|
||||||
AppId={{21E8D429-0C18-462F-AFC0-56EA664DE629}
|
AppId={{21E8D429-0C18-462F-AFC0-56EA664DE629}
|
||||||
|
|||||||
17
setup.py
17
setup.py
@@ -1,14 +1,16 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
"""Setupscript for RevPiCommander."""
|
"""Setup script for RevPiCommander."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2018 Sven Sager"
|
__copyright__ = "Copyright (C) 2018 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
from setuptools import find_namespace_packages, setup
|
from setuptools import find_namespace_packages, setup
|
||||||
|
|
||||||
|
from src.revpicommander import __version__
|
||||||
|
|
||||||
setup(
|
setup(
|
||||||
name="revpicommander",
|
name="revpicommander",
|
||||||
version="0.9.10rc2",
|
version=__version__,
|
||||||
|
|
||||||
packages=find_namespace_packages("src"),
|
packages=find_namespace_packages("src"),
|
||||||
package_dir={'': 'src'},
|
package_dir={'': 'src'},
|
||||||
@@ -17,31 +19,32 @@ setup(
|
|||||||
install_requires=[
|
install_requires=[
|
||||||
"keyring",
|
"keyring",
|
||||||
"PyQt5",
|
"PyQt5",
|
||||||
|
"paramiko",
|
||||||
"revpimodio2",
|
"revpimodio2",
|
||||||
"zeroconf"
|
"zeroconf"
|
||||||
],
|
],
|
||||||
entry_points={
|
entry_points={
|
||||||
'gui_scripts': [
|
'gui_scripts': [
|
||||||
'revpicommander-gui = revpicommander.revpicommander:main',
|
'revpicommander = revpicommander.revpicommander:main',
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
|
||||||
platforms=["all"],
|
platforms=["all"],
|
||||||
|
|
||||||
url="https://revpimodio.org/revpipyplc/",
|
url="https://revpimodio.org/revpipyplc/",
|
||||||
license="GPLv3",
|
license="GPLv2",
|
||||||
author="Sven Sager",
|
author="Sven Sager",
|
||||||
author_email="akira@narux.de",
|
author_email="akira@narux.de",
|
||||||
maintainer="Sven Sager",
|
maintainer="Sven Sager",
|
||||||
maintainer_email="akira@revpimodio.org",
|
maintainer_email="akira@revpimodio.org",
|
||||||
description="GUI for Revolution Pi to upload programs and do IO-Checks",
|
description="GUI for Revolution Pi to upload programs and do IO-Checks",
|
||||||
long_description="The RevPiCommander is a GUI tool to manage your revolution Pi over the\n"
|
long_description="The RevPiCommander is a GUI tool to manage your Revolution Pi over the\n"
|
||||||
"network. You can search for RevPis in your network, manage the settings\n"
|
"network. You can search for RevPis in your network, manage the settings\n"
|
||||||
"of RevPiPyLoad and do IO checks on your local machine. Developing your\n"
|
"of RevPiPyLoad and do IO checks on your local machine. Developing your\n"
|
||||||
"control program is very easy with the developer, upload and debug it\n"
|
"control program is very easy with the developer, upload and debug it\n"
|
||||||
"over the network.",
|
"over the network.",
|
||||||
keywords=["revpi", "revolution pi", "revpimodio", "plc"],
|
keywords=["revpi", "revolution pi", "revpimodio", "plc"],
|
||||||
classifiers=[
|
classifiers=[
|
||||||
"License :: OSI Approved :: GNU General Public License v3 (GPLv3)",
|
"License :: OSI Approved :: GNU General Public License v2 (GPLv2)",
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -2,4 +2,6 @@
|
|||||||
"""Package: RevPiCommander."""
|
"""Package: RevPiCommander."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
__package__ = "revpicommander"
|
||||||
|
__version__ = "0.10.0"
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""Start main application of this package."""
|
"""Start main application of this package."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
# If we are running from a wheel, add the wheel to sys.path
|
# If we are running from a wheel, add the wheel to sys.path
|
||||||
if __package__ == "":
|
if __package__ == "":
|
||||||
@@ -16,6 +16,14 @@ if __package__ == "":
|
|||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
|
if len(sys.argv) == 2 and "--version" in sys.argv:
|
||||||
|
# Catch --version, if this is the only argument (sys.argv[0] is always the script name)
|
||||||
|
from revpicommander import __version__
|
||||||
|
print(__version__)
|
||||||
|
sys.exit(0)
|
||||||
|
|
||||||
|
else:
|
||||||
from revpicommander.revpicommander import main
|
from revpicommander.revpicommander import main
|
||||||
|
|
||||||
# Run the main application of this package
|
# Run the main application of this package
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""Manager for ACL lists."""
|
"""Manager for ACL lists."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
from re import compile
|
from re import compile
|
||||||
|
|
||||||
@@ -16,7 +16,7 @@ class AclManager(QtWidgets.QDialog, Ui_diag_aclmanager):
|
|||||||
"""ACL manager."""
|
"""ACL manager."""
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
super(AclManager, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
self.setFixedSize(self.size())
|
self.setFixedSize(self.size())
|
||||||
|
|
||||||
@@ -91,7 +91,7 @@ class AclManager(QtWidgets.QDialog, Ui_diag_aclmanager):
|
|||||||
return
|
return
|
||||||
|
|
||||||
self.__oldacl = self.__table_to_acl()
|
self.__oldacl = self.__table_to_acl()
|
||||||
super(AclManager, self).accept()
|
super().accept()
|
||||||
|
|
||||||
def closeEvent(self, a0: QtGui.QCloseEvent) -> None:
|
def closeEvent(self, a0: QtGui.QCloseEvent) -> None:
|
||||||
if self._changes_done():
|
if self._changes_done():
|
||||||
@@ -107,12 +107,12 @@ class AclManager(QtWidgets.QDialog, Ui_diag_aclmanager):
|
|||||||
a0.ignore()
|
a0.ignore()
|
||||||
|
|
||||||
def exec(self) -> int:
|
def exec(self) -> int:
|
||||||
return super(AclManager, self).exec()
|
return super().exec()
|
||||||
|
|
||||||
def reject(self) -> None:
|
def reject(self) -> None:
|
||||||
"""Restore old settings."""
|
"""Restore old settings."""
|
||||||
self.setup_acl_manager(self.__oldacl, self.__dict_acltext)
|
self.setup_acl_manager(self.__oldacl, self.__dict_acltext)
|
||||||
super(AclManager, self).reject()
|
super().reject()
|
||||||
|
|
||||||
def setup_acl_manager(self, acl_string: str, acl_texts: dict):
|
def setup_acl_manager(self, acl_string: str, acl_texts: dict):
|
||||||
if type(acl_string) != str:
|
if type(acl_string) != str:
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""Revolution Pi search with zeroconf."""
|
"""Revolution Pi search with zeroconf."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
import webbrowser
|
import webbrowser
|
||||||
from re import compile
|
from re import compile
|
||||||
@@ -24,7 +24,7 @@ class AvahiSearchThread(QtCore.QThread):
|
|||||||
updated = QtCore.pyqtSignal(str, str, int, str, str)
|
updated = QtCore.pyqtSignal(str, str, int, str, str)
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
super(AvahiSearchThread, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self._cycle_wait_ms = 1000
|
self._cycle_wait_ms = 1000
|
||||||
|
|
||||||
self.re_posix = compile(
|
self.re_posix = compile(
|
||||||
@@ -71,7 +71,7 @@ class AvahiSearchThread(QtCore.QThread):
|
|||||||
class AvahiSearch(QtWidgets.QDialog, Ui_diag_search):
|
class AvahiSearch(QtWidgets.QDialog, Ui_diag_search):
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
super(AvahiSearch, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
|
|
||||||
# Global variables to let parent decide other actions
|
# Global variables to let parent decide other actions
|
||||||
@@ -130,7 +130,7 @@ class AvahiSearch(QtWidgets.QDialog, Ui_diag_search):
|
|||||||
self.act_connect_xmlrpc.setVisible(True)
|
self.act_connect_xmlrpc.setVisible(True)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _find_settings(address: str) -> list[RevPiSettings]:
|
def _find_settings(address: str):
|
||||||
"""Find all settings with known avahi_id."""
|
"""Find all settings with known avahi_id."""
|
||||||
return [
|
return [
|
||||||
revpi_setting
|
revpi_setting
|
||||||
@@ -184,7 +184,7 @@ class AvahiSearch(QtWidgets.QDialog, Ui_diag_search):
|
|||||||
self.connect_settings = None
|
self.connect_settings = None
|
||||||
self.just_save = False
|
self.just_save = False
|
||||||
self._restart_search()
|
self._restart_search()
|
||||||
rc = super(AvahiSearch, self).exec()
|
rc = super().exec()
|
||||||
self._th_zero_conf.requestInterruption()
|
self._th_zero_conf.requestInterruption()
|
||||||
return rc
|
return rc
|
||||||
|
|
||||||
@@ -258,8 +258,11 @@ class AvahiSearch(QtWidgets.QDialog, Ui_diag_search):
|
|||||||
return
|
return
|
||||||
item = selected_items[0]
|
item = selected_items[0]
|
||||||
|
|
||||||
# Till we could not choose https / http we are using the ip address
|
# We should use the hostname on macOS to let safari connect in link local mode (for linux nice too)
|
||||||
webbrowser.open("http://{0}/".format(item.data(WidgetData.address)))
|
webbrowser.open("http://{0}/".format(item.data(
|
||||||
|
WidgetData.address if platform == "win32"
|
||||||
|
else WidgetData.host_name_full
|
||||||
|
)))
|
||||||
|
|
||||||
@QtCore.pyqtSlot(str, str, int, str, str)
|
@QtCore.pyqtSlot(str, str, int, str, str)
|
||||||
def on_avahi_added(self, avahi_id: str, server: str, port: int, conf_type: str, ip: str) -> None:
|
def on_avahi_added(self, avahi_id: str, server: str, port: int, conf_type: str, ip: str) -> None:
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""File transfer system to handle QThreads."""
|
"""File transfer system to handle QThreads."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
from logging import getLogger
|
from logging import getLogger
|
||||||
|
|
||||||
@@ -18,8 +18,9 @@ class BackgroundWorker(QtCore.QThread):
|
|||||||
steps_done = QtCore.pyqtSignal(int)
|
steps_done = QtCore.pyqtSignal(int)
|
||||||
status_message = QtCore.pyqtSignal(str)
|
status_message = QtCore.pyqtSignal(str)
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None, interruption_text: str = None):
|
||||||
super(BackgroundWorker, self).__init__(parent)
|
super().__init__(parent)
|
||||||
|
self._interruption_text = interruption_text or self.tr("User requested cancellation...")
|
||||||
|
|
||||||
def check_cancel(self) -> bool:
|
def check_cancel(self) -> bool:
|
||||||
"""
|
"""
|
||||||
@@ -28,33 +29,67 @@ class BackgroundWorker(QtCore.QThread):
|
|||||||
:return: True, if interruption was requested
|
:return: True, if interruption was requested
|
||||||
"""
|
"""
|
||||||
if self.isInterruptionRequested():
|
if self.isInterruptionRequested():
|
||||||
self.status_message.emit(self.tr("User requested cancellation..."))
|
self.status_message.emit(self._interruption_text)
|
||||||
self.msleep(750)
|
self.msleep(750)
|
||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def exec_dialog(self) -> int:
|
def exec_dialog(self, window_title="", can_cancel=True) -> int:
|
||||||
|
"""
|
||||||
|
Show dialog with progress bar.
|
||||||
|
|
||||||
|
:param window_title: Title of Dialog window
|
||||||
|
:param can_cancel: If False, the cancel button is deactivated
|
||||||
|
:return: Dialog result
|
||||||
|
"""
|
||||||
diag = WorkerDialog(self, self.parent())
|
diag = WorkerDialog(self, self.parent())
|
||||||
|
diag.setWindowTitle(window_title)
|
||||||
|
diag.btn_box.setEnabled(can_cancel)
|
||||||
rc = diag.exec()
|
rc = diag.exec()
|
||||||
diag.deleteLater()
|
diag.deleteLater()
|
||||||
return rc
|
return rc
|
||||||
|
|
||||||
def wait_interruptable(self, seconds=-1) -> None:
|
def wait_interruptable(self, seconds=-1) -> bool:
|
||||||
"""Save function to wait and get the cancel buttons."""
|
"""
|
||||||
|
Save function to wait and get the cancel buttons.
|
||||||
|
|
||||||
|
:param seconds: Wait this amount of seconds
|
||||||
|
:return: True, if interruption was requested
|
||||||
|
"""
|
||||||
counter = seconds * 4
|
counter = seconds * 4
|
||||||
while counter != 0:
|
while counter != 0:
|
||||||
counter -= 1
|
counter -= 1
|
||||||
self.msleep(250)
|
self.msleep(250)
|
||||||
if self.check_cancel():
|
if self.check_cancel():
|
||||||
break
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
def run(self) -> None:
|
def run(self) -> None:
|
||||||
"""Worker thread to import pictures from camera."""
|
"""Override this function with your logic."""
|
||||||
log.debug("BackgroundWorker.run")
|
raise NotImplementedError()
|
||||||
self.status_message.emit("Started dummy thread...")
|
|
||||||
self.wait_interruptable(5)
|
|
||||||
self.status_message.emit("Completed dummy thread.")
|
class BackgroundWaiter(BackgroundWorker):
|
||||||
self.wait_interruptable(2)
|
"""Just wait an amount of time and show progress bar."""
|
||||||
|
|
||||||
|
def __init__(self, seconds: int, status_message: str, parent=None, interruption_text: str = None):
|
||||||
|
super().__init__(parent, interruption_text)
|
||||||
|
self._status_message = status_message
|
||||||
|
self._wait_steps = seconds * 4
|
||||||
|
|
||||||
|
def run(self) -> None:
|
||||||
|
log.debug("BackgroundWaiter.run")
|
||||||
|
self.steps_todo.emit(self._wait_steps)
|
||||||
|
self.status_message.emit(self._status_message)
|
||||||
|
counter = 0
|
||||||
|
while counter <= self._wait_steps:
|
||||||
|
counter += 1
|
||||||
|
self.msleep(250)
|
||||||
|
if self.isInterruptionRequested():
|
||||||
|
self.steps_done.emit(self._wait_steps)
|
||||||
|
if self.check_cancel():
|
||||||
|
return
|
||||||
|
self.steps_done.emit(counter)
|
||||||
|
|
||||||
|
|
||||||
class WorkerDialog(QtWidgets.QDialog, Ui_diag_backgroundworker):
|
class WorkerDialog(QtWidgets.QDialog, Ui_diag_backgroundworker):
|
||||||
@@ -66,7 +101,7 @@ class WorkerDialog(QtWidgets.QDialog, Ui_diag_backgroundworker):
|
|||||||
:param worker_thread: Thread with the logic work to do
|
:param worker_thread: Thread with the logic work to do
|
||||||
:param parent: QtWidget
|
:param parent: QtWidget
|
||||||
"""
|
"""
|
||||||
super(WorkerDialog, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
|
|
||||||
self._canceled = False
|
self._canceled = False
|
||||||
@@ -82,7 +117,7 @@ class WorkerDialog(QtWidgets.QDialog, Ui_diag_backgroundworker):
|
|||||||
|
|
||||||
def exec(self) -> int:
|
def exec(self) -> int:
|
||||||
self._th.start()
|
self._th.start()
|
||||||
return super(WorkerDialog, self).exec()
|
return super().exec()
|
||||||
|
|
||||||
@QtCore.pyqtSlot()
|
@QtCore.pyqtSlot()
|
||||||
def on_th_finished(self) -> None:
|
def on_th_finished(self) -> None:
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""Debug control widget to append to main window."""
|
"""Debug control widget to append to main window."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
import pickle
|
import pickle
|
||||||
from xmlrpc.client import Binary, Fault, MultiCall, MultiCallIterator
|
from xmlrpc.client import Binary, Fault, MultiCall, MultiCallIterator
|
||||||
@@ -26,7 +26,7 @@ class PsValues(QtCore.QThread):
|
|||||||
process_image_received = QtCore.pyqtSignal(Binary)
|
process_image_received = QtCore.pyqtSignal(Binary)
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
super(PsValues, self).__init__()
|
super().__init__()
|
||||||
self._cycle_time = 200
|
self._cycle_time = 200
|
||||||
|
|
||||||
def run(self):
|
def run(self):
|
||||||
@@ -55,7 +55,7 @@ class DebugControl(QtWidgets.QWidget, Ui_wid_debugcontrol):
|
|||||||
"""Debug controller for main window."""
|
"""Debug controller for main window."""
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
super(DebugControl, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
|
|
||||||
self.dict_devices = {}
|
self.dict_devices = {}
|
||||||
@@ -197,6 +197,7 @@ class DebugControl(QtWidgets.QWidget, Ui_wid_debugcontrol):
|
|||||||
win = self.dict_windows[position]
|
win = self.dict_windows[position]
|
||||||
for io in self.dict_ios[io_type][position]: # type: list
|
for io in self.dict_ios[io_type][position]: # type: list
|
||||||
# ['name', bytelen, byte_address, 'bmk', bitaddress, 'byteorder', signed]
|
# ['name', bytelen, byte_address, 'bmk', bitaddress, 'byteorder', signed]
|
||||||
|
# + wordorder since revpipyload 0.9.9
|
||||||
value_procimg = bytes(ba_values[io[2]:io[2] + io[1]])
|
value_procimg = bytes(ba_values[io[2]:io[2] + io[1]])
|
||||||
if io[4] >= 0:
|
if io[4] >= 0:
|
||||||
# Bit-IO
|
# Bit-IO
|
||||||
@@ -283,7 +284,7 @@ class DebugControl(QtWidgets.QWidget, Ui_wid_debugcontrol):
|
|||||||
self.cbx_refresh.setChecked(False)
|
self.cbx_refresh.setChecked(False)
|
||||||
self._destroy_io_view()
|
self._destroy_io_view()
|
||||||
|
|
||||||
super(DebugControl, self).deleteLater()
|
super().deleteLater()
|
||||||
|
|
||||||
def reload_devices(self):
|
def reload_devices(self):
|
||||||
"""Rebuild GUI depending on devices and ios of Revolution Pi."""
|
"""Rebuild GUI depending on devices and ios of Revolution Pi."""
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""One device of the Revolution Pi."""
|
"""One device of the Revolution Pi."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
import struct
|
import struct
|
||||||
|
|
||||||
@@ -22,7 +22,7 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
search_class = (QtWidgets.QLineEdit, QtWidgets.QDoubleSpinBox, QtWidgets.QCheckBox)
|
search_class = (QtWidgets.QLineEdit, QtWidgets.QDoubleSpinBox, QtWidgets.QCheckBox)
|
||||||
|
|
||||||
def __init__(self, position: int, name: str, inputs: list, outputs: list, parent=None):
|
def __init__(self, position: int, name: str, inputs: list, outputs: list, parent=None):
|
||||||
super(DebugIos, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
|
|
||||||
self.restoreGeometry(helper.cm.settings.debug_geos.get(position, b''))
|
self.restoreGeometry(helper.cm.settings.debug_geos.get(position, b''))
|
||||||
@@ -38,9 +38,7 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
self.outputs = outputs.copy()
|
self.outputs = outputs.copy()
|
||||||
self.write_values = False
|
self.write_values = False
|
||||||
|
|
||||||
min_input = min(inputs, key=lambda k: k[2])
|
self.length = self._calc_device_length(self.inputs, self.outputs)
|
||||||
max_output = max(outputs, key=lambda k: k[2])
|
|
||||||
self.length = max_output[2] + max_output[1] - min_input[2]
|
|
||||||
|
|
||||||
self.style_sheet = ""
|
self.style_sheet = ""
|
||||||
self._create_io(self.inputs, self.saw_inp, True)
|
self._create_io(self.inputs, self.saw_inp, True)
|
||||||
@@ -62,6 +60,23 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
return max_int_value / 2 * -1 if signed else 0.0, \
|
return max_int_value / 2 * -1 if signed else 0.0, \
|
||||||
max_int_value / 2 - 1 if signed else max_int_value - 1
|
max_int_value / 2 - 1 if signed else max_int_value - 1
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _calc_device_length(inputs: list, outputs: list) -> int:
|
||||||
|
"""Calculate the device length with IO data."""
|
||||||
|
if inputs and outputs:
|
||||||
|
min_input = min(inputs, key=lambda k: k[2])
|
||||||
|
max_output = max(outputs, key=lambda k: k[2])
|
||||||
|
elif inputs:
|
||||||
|
min_input = min(inputs, key=lambda k: k[2])
|
||||||
|
max_output = max(inputs, key=lambda k: k[2])
|
||||||
|
elif outputs:
|
||||||
|
min_input = min(outputs, key=lambda k: k[2])
|
||||||
|
max_output = max(outputs, key=lambda k: k[2])
|
||||||
|
else:
|
||||||
|
return 0
|
||||||
|
|
||||||
|
return max_output[2] + max_output[1] - min_input[2]
|
||||||
|
|
||||||
def _create_io(self, lst_ios: list, container: QtWidgets.QWidget, read_only: bool):
|
def _create_io(self, lst_ios: list, container: QtWidgets.QWidget, read_only: bool):
|
||||||
lst_names = list(lst[0] for lst in lst_ios)
|
lst_names = list(lst[0] for lst in lst_ios)
|
||||||
layout = container.layout() # type: QtWidgets.QFormLayout
|
layout = container.layout() # type: QtWidgets.QFormLayout
|
||||||
@@ -82,6 +97,7 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
bit_address = io[4]
|
bit_address = io[4]
|
||||||
byteorder = io[5]
|
byteorder = io[5]
|
||||||
signed = io[6]
|
signed = io[6]
|
||||||
|
word_order = io[7] if len(io) > 7 else "ignored"
|
||||||
|
|
||||||
val = container.findChild(self.search_class, name)
|
val = container.findChild(self.search_class, name)
|
||||||
if val is not None:
|
if val is not None:
|
||||||
@@ -100,14 +116,15 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
lbl.setObjectName("lbl_".format(name))
|
lbl.setObjectName("lbl_".format(name))
|
||||||
lbl.setStyleSheet(self.style_sheet)
|
lbl.setStyleSheet(self.style_sheet)
|
||||||
|
|
||||||
val = self._create_widget(name, byte_length, bit_address, byteorder, signed, read_only)
|
val = self._create_widget(name, byte_length, bit_address, byteorder, signed, read_only, word_order)
|
||||||
val.setParent(container)
|
val.setParent(container)
|
||||||
layout.insertRow(counter, val, lbl)
|
layout.insertRow(counter, val, lbl)
|
||||||
|
|
||||||
self.splitter.setSizes([1, 1])
|
self.splitter.setSizes([1, 1])
|
||||||
|
|
||||||
def _create_widget(
|
def _create_widget(
|
||||||
self, name: str, byte_length: int, bit_address: int, byteorder: str, signed: bool, read_only: bool):
|
self, name: str, byte_length: int, bit_address: int, byteorder: str, signed: bool, read_only: bool,
|
||||||
|
word_order: str):
|
||||||
"""Create widget in functions address space to use lambda functions."""
|
"""Create widget in functions address space to use lambda functions."""
|
||||||
if bit_address >= 0:
|
if bit_address >= 0:
|
||||||
val = QtWidgets.QCheckBox()
|
val = QtWidgets.QCheckBox()
|
||||||
@@ -160,6 +177,7 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
val.setProperty("bit_address", bit_address)
|
val.setProperty("bit_address", bit_address)
|
||||||
val.setProperty("byte_length", byte_length)
|
val.setProperty("byte_length", byte_length)
|
||||||
val.setProperty("signed", signed)
|
val.setProperty("signed", signed)
|
||||||
|
val.setProperty("word_order", word_order)
|
||||||
|
|
||||||
self.__qwa[name] = val
|
self.__qwa[name] = val
|
||||||
return val
|
return val
|
||||||
@@ -203,6 +221,9 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
act_as_number = QtWidgets.QAction(self.tr("as number"))
|
act_as_number = QtWidgets.QAction(self.tr("as number"))
|
||||||
men.addAction(act_as_number)
|
men.addAction(act_as_number)
|
||||||
men.addSeparator()
|
men.addSeparator()
|
||||||
|
else:
|
||||||
|
act_as_text = None
|
||||||
|
act_as_number = None
|
||||||
|
|
||||||
act_signed = QtWidgets.QAction(self.tr("signed"), men)
|
act_signed = QtWidgets.QAction(self.tr("signed"), men)
|
||||||
act_signed.setCheckable(True)
|
act_signed.setCheckable(True)
|
||||||
@@ -214,6 +235,14 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
act_byteorder.setChecked(sender.property("big_endian") or False)
|
act_byteorder.setChecked(sender.property("big_endian") or False)
|
||||||
men.addAction(act_byteorder)
|
men.addAction(act_byteorder)
|
||||||
|
|
||||||
|
if sender.property("byte_length") > 2:
|
||||||
|
act_wordorder = QtWidgets.QAction(self.tr("switch wordorder"))
|
||||||
|
act_wordorder.setCheckable(True)
|
||||||
|
act_wordorder.setChecked(sender.property("word_order") == "big")
|
||||||
|
men.addAction(act_wordorder)
|
||||||
|
else:
|
||||||
|
act_wordorder = None
|
||||||
|
|
||||||
rc = men.exec(sender.mapToGlobal(point))
|
rc = men.exec(sender.mapToGlobal(point))
|
||||||
if not rc:
|
if not rc:
|
||||||
men.deleteLater()
|
men.deleteLater()
|
||||||
@@ -232,6 +261,8 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
sender.setMaximum(max_value)
|
sender.setMaximum(max_value)
|
||||||
elif rc == act_byteorder:
|
elif rc == act_byteorder:
|
||||||
sender.setProperty("big_endian", act_byteorder.isChecked())
|
sender.setProperty("big_endian", act_byteorder.isChecked())
|
||||||
|
elif rc == act_wordorder:
|
||||||
|
sender.setProperty("word_order", "big" if act_wordorder.isChecked() else "little")
|
||||||
|
|
||||||
if sender.property("frm"):
|
if sender.property("frm"):
|
||||||
sender.setProperty("frm", "{0}{1}".format(
|
sender.setProperty("frm", "{0}{1}".format(
|
||||||
@@ -239,8 +270,7 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
sender.property("struct_type").lower() if act_signed.isChecked()
|
sender.property("struct_type").lower() if act_signed.isChecked()
|
||||||
else sender.property("struct_type").upper()
|
else sender.property("struct_type").upper()
|
||||||
))
|
))
|
||||||
elif sender.property("byte_length") > 4:
|
elif rc == act_as_text:
|
||||||
if rc == act_as_text:
|
|
||||||
sender.setProperty("struct_type", "text")
|
sender.setProperty("struct_type", "text")
|
||||||
elif rc == act_as_number:
|
elif rc == act_as_number:
|
||||||
sender.setProperty("struct_type", "number")
|
sender.setProperty("struct_type", "number")
|
||||||
@@ -278,10 +308,7 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
"""Update IOs after driver reset of piCtory."""
|
"""Update IOs after driver reset of piCtory."""
|
||||||
|
|
||||||
# Check device length, this has to match to reuse this device
|
# Check device length, this has to match to reuse this device
|
||||||
min_input = min(inputs, key=lambda k: k[2])
|
if self.length != self._calc_device_length(inputs, outputs):
|
||||||
max_output = max(outputs, key=lambda k: k[2])
|
|
||||||
new_length = max_output[2] + max_output[1] - min_input[2]
|
|
||||||
if self.length != new_length:
|
|
||||||
return False
|
return False
|
||||||
|
|
||||||
# Remove IOs, which was remove or renamed
|
# Remove IOs, which was remove or renamed
|
||||||
@@ -335,6 +362,10 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
:param just_last_value: Just set last value property
|
:param just_last_value: Just set last value property
|
||||||
"""
|
"""
|
||||||
child = self.__qwa[io_name]
|
child = self.__qwa[io_name]
|
||||||
|
|
||||||
|
if child.property("word_order") == "big" and type(value) == bytes:
|
||||||
|
value = helper.swap_word_order(value)
|
||||||
|
|
||||||
if child.property("frm"):
|
if child.property("frm"):
|
||||||
value = struct.unpack(child.property("frm"), value)[0]
|
value = struct.unpack(child.property("frm"), value)[0]
|
||||||
elif type(value) == bytes:
|
elif type(value) == bytes:
|
||||||
@@ -349,6 +380,7 @@ class DebugIos(QtWidgets.QMainWindow, Ui_win_debugios):
|
|||||||
).format(value, io_name)
|
).format(value, io_name)
|
||||||
)
|
)
|
||||||
if child.property("struct_type") == "number":
|
if child.property("struct_type") == "number":
|
||||||
|
# fixme: Crashs with too much bytes
|
||||||
value = str(int.from_bytes(
|
value = str(int.from_bytes(
|
||||||
value,
|
value,
|
||||||
byteorder="big" if child.property("big_endian") else "little",
|
byteorder="big" if child.property("big_endian") else "little",
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""Helper functions for this application."""
|
"""Helper functions for this application."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
import pickle
|
import pickle
|
||||||
import socket
|
import socket
|
||||||
@@ -11,24 +11,31 @@ from http.client import CannotSendRequest
|
|||||||
from os import environ, remove
|
from os import environ, remove
|
||||||
from os.path import exists
|
from os.path import exists
|
||||||
from queue import Queue
|
from queue import Queue
|
||||||
|
from re import search
|
||||||
from threading import Lock
|
from threading import Lock
|
||||||
from uuid import uuid4
|
from uuid import uuid4
|
||||||
from xmlrpc.client import Binary, ServerProxy
|
from xmlrpc.client import Binary, ServerProxy
|
||||||
|
|
||||||
from PyQt5 import QtCore, QtWidgets
|
from PyQt5 import QtCore
|
||||||
from paramiko.ssh_exception import AuthenticationException
|
from paramiko.ssh_exception import AuthenticationException
|
||||||
|
|
||||||
from . import proginit as pi
|
from . import proginit as pi
|
||||||
from .ssh_tunneling.server import SSHLocalTunnel
|
from .ssh_tunneling.server import SSHLocalTunnel
|
||||||
from .sshauth import SSHAuth
|
|
||||||
|
|
||||||
settings = QtCore.QSettings("revpimodio.org", "RevPiCommander")
|
settings = QtCore.QSettings("revpimodio.org", "revpicommander")
|
||||||
"""Global application settings."""
|
"""Global application settings."""
|
||||||
|
|
||||||
homedir = environ.get("HOME", "") or environ.get("APPDATA", "")
|
homedir = environ.get("HOME", "") or environ.get("APPDATA", "")
|
||||||
"""Home dir of user."""
|
"""Home dir of user."""
|
||||||
|
|
||||||
|
|
||||||
|
class ConnectionFail(IntEnum):
|
||||||
|
NO_XML_RPC = 1
|
||||||
|
SSH_CONNECT = 2
|
||||||
|
SSH_AUTH = 4
|
||||||
|
NO_XML_RPC_VIA_TUNNEL = 9 # Includes NO_XML_RPC Bit
|
||||||
|
|
||||||
|
|
||||||
class WidgetData(IntEnum):
|
class WidgetData(IntEnum):
|
||||||
address = 260
|
address = 260
|
||||||
acl_level = 262
|
acl_level = 262
|
||||||
@@ -194,6 +201,8 @@ class RevPiSettings:
|
|||||||
class ConnectionManager(QtCore.QThread):
|
class ConnectionManager(QtCore.QThread):
|
||||||
"""Check connection and status for PLC program on Revolution Pi."""
|
"""Check connection and status for PLC program on Revolution Pi."""
|
||||||
|
|
||||||
|
connect_error = QtCore.pyqtSignal(str, str, ConnectionFail, RevPiSettings)
|
||||||
|
"""Error header, message and reason (ConnectionFail) of a new connection after pyload_connect call."""
|
||||||
connection_established = QtCore.pyqtSignal()
|
connection_established = QtCore.pyqtSignal()
|
||||||
"""New connection established successfully with <class 'ServerProxy'>."""
|
"""New connection established successfully with <class 'ServerProxy'>."""
|
||||||
connection_disconnected = QtCore.pyqtSignal()
|
connection_disconnected = QtCore.pyqtSignal()
|
||||||
@@ -208,7 +217,7 @@ class ConnectionManager(QtCore.QThread):
|
|||||||
"""After errors the connection is established again, could have other port information (SSH)."""
|
"""After errors the connection is established again, could have other port information (SSH)."""
|
||||||
|
|
||||||
def __init__(self, parent=None, cycle_time_ms=1000):
|
def __init__(self, parent=None, cycle_time_ms=1000):
|
||||||
super(ConnectionManager, self).__init__(parent)
|
super().__init__(parent)
|
||||||
|
|
||||||
self._cli = None
|
self._cli = None
|
||||||
self._cli_connect = Queue()
|
self._cli_connect = Queue()
|
||||||
@@ -225,6 +234,8 @@ class ConnectionManager(QtCore.QThread):
|
|||||||
|
|
||||||
self.pyload_version = (0, 0, 0)
|
self.pyload_version = (0, 0, 0)
|
||||||
"""Version number of RevPiPyLoad 0.0.0 with <class 'int'> values."""
|
"""Version number of RevPiPyLoad 0.0.0 with <class 'int'> values."""
|
||||||
|
self.pyload_version_str = ""
|
||||||
|
"""Raw string of RevPyPyLoad version, could contain rc1 at the end."""
|
||||||
self.xml_funcs = []
|
self.xml_funcs = []
|
||||||
"""Name list of all supported functions of RevPiPyLoad."""
|
"""Name list of all supported functions of RevPiPyLoad."""
|
||||||
self.xml_mode = -1
|
self.xml_mode = -1
|
||||||
@@ -308,15 +319,16 @@ class ConnectionManager(QtCore.QThread):
|
|||||||
self.ssh_pass = ""
|
self.ssh_pass = ""
|
||||||
|
|
||||||
self.pyload_version = (0, 0, 0)
|
self.pyload_version = (0, 0, 0)
|
||||||
|
self.pyload_version_str = ""
|
||||||
self.xml_funcs.clear()
|
self.xml_funcs.clear()
|
||||||
self.xml_mode = -1
|
self.xml_mode = -1
|
||||||
|
|
||||||
def pyload_connect(self, revpi_settings: RevPiSettings, parent=None) -> bool:
|
def pyload_connect(self, revpi_settings: RevPiSettings, ssh_pass="") -> bool:
|
||||||
"""
|
"""
|
||||||
Create a new connection from settings object.
|
Create a new connection from settings object.
|
||||||
|
|
||||||
:param revpi_settings: Revolution Pi saved connection settings
|
:param revpi_settings: Revolution Pi saved connection settings
|
||||||
:param parent: Qt parent window for dialog positioning
|
:param ssh_pass: Use this ssh password, if revpi_settings.ssh_use_tunnel is true
|
||||||
:return: True, if the connection was successfully established
|
:return: True, if the connection was successfully established
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -325,50 +337,40 @@ class ConnectionManager(QtCore.QThread):
|
|||||||
|
|
||||||
ssh_tunnel_server = None
|
ssh_tunnel_server = None
|
||||||
ssh_tunnel_port = 0
|
ssh_tunnel_port = 0
|
||||||
ssh_pass = ""
|
|
||||||
|
|
||||||
socket.setdefaulttimeout(2)
|
socket.setdefaulttimeout(revpi_settings.timeout)
|
||||||
|
|
||||||
if revpi_settings.ssh_use_tunnel:
|
if revpi_settings.ssh_use_tunnel:
|
||||||
while True:
|
|
||||||
diag_ssh_auth = SSHAuth(
|
|
||||||
revpi_settings.ssh_user,
|
|
||||||
"{0}.{1}_{2}".format(
|
|
||||||
settings.applicationName(),
|
|
||||||
settings.organizationName(),
|
|
||||||
revpi_settings.internal_id),
|
|
||||||
parent,
|
|
||||||
)
|
|
||||||
if not diag_ssh_auth.exec() == QtWidgets.QDialog.Accepted:
|
|
||||||
self._clear_settings()
|
|
||||||
return False
|
|
||||||
|
|
||||||
ssh_user = diag_ssh_auth.username
|
|
||||||
ssh_pass = diag_ssh_auth.password
|
|
||||||
ssh_tunnel_server = SSHLocalTunnel(
|
ssh_tunnel_server = SSHLocalTunnel(
|
||||||
revpi_settings.port,
|
revpi_settings.port,
|
||||||
revpi_settings.address,
|
revpi_settings.address,
|
||||||
revpi_settings.ssh_port
|
revpi_settings.ssh_port
|
||||||
)
|
)
|
||||||
revpi_settings.ssh_saved_password = diag_ssh_auth.in_keyring
|
|
||||||
try:
|
try:
|
||||||
ssh_tunnel_port = ssh_tunnel_server.connect_by_credentials(ssh_user, ssh_pass)
|
ssh_tunnel_port = ssh_tunnel_server.connect_by_credentials(revpi_settings.ssh_user, ssh_pass)
|
||||||
break
|
|
||||||
|
if getattr(revpi_settings, "ssh_enable_revpipyload", False):
|
||||||
|
ssh_tunnel_server.send_cmd("sudo systemctl enable --now revpipyload")
|
||||||
|
|
||||||
except AuthenticationException:
|
except AuthenticationException:
|
||||||
diag_ssh_auth.remove_saved_password()
|
self.connect_error.emit(
|
||||||
QtWidgets.QMessageBox.critical(
|
self.tr("Error"), self.tr(
|
||||||
parent, self.tr("Error"), self.tr(
|
|
||||||
"The combination of username and password was rejected from the SSH server.\n\n"
|
"The combination of username and password was rejected from the SSH server.\n\n"
|
||||||
"Try again."
|
"Try again."
|
||||||
|
),
|
||||||
|
ConnectionFail.SSH_AUTH,
|
||||||
|
revpi_settings,
|
||||||
)
|
)
|
||||||
)
|
return False
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
# todo: Check some more kinds of exceptions and nice user info
|
# todo: Check some more kinds of exceptions and nice user info
|
||||||
self._clear_settings()
|
self._clear_settings()
|
||||||
QtWidgets.QMessageBox.critical(
|
self.connect_error.emit(
|
||||||
parent, self.tr("Error"), self.tr(
|
self.tr("Error"), self.tr(
|
||||||
"Could not establish a SSH connection to server:\n\n{0}"
|
"Could not establish a SSH connection to server:\n\n{0}"
|
||||||
).format(str(e))
|
).format(str(e)),
|
||||||
|
ConnectionFail.SSH_CONNECT,
|
||||||
|
revpi_settings,
|
||||||
)
|
)
|
||||||
return False
|
return False
|
||||||
|
|
||||||
@@ -379,24 +381,41 @@ class ConnectionManager(QtCore.QThread):
|
|||||||
|
|
||||||
# Load values and test connection to Revolution Pi
|
# Load values and test connection to Revolution Pi
|
||||||
try:
|
try:
|
||||||
pyload_version = tuple(map(int, sp.version().split(".")))
|
ma = search(r"(?P<major>\d+)\.(?P<minor>\d+)\.(?P<patch>\d+)", sp.version())
|
||||||
|
pyload_version = int(ma.group("major")), int(ma.group("minor")), int(ma.group("patch"))
|
||||||
|
pyload_version_str = ma.string
|
||||||
xml_funcs = sp.system.listMethods()
|
xml_funcs = sp.system.listMethods()
|
||||||
xml_mode = sp.xmlmodus()
|
xml_mode = sp.xmlmodus()
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
pi.logger.exception(e)
|
pi.logger.exception(e)
|
||||||
self.connection_error_observed.emit(str(e))
|
self.connection_error_observed.emit(str(e))
|
||||||
|
|
||||||
if not revpi_settings.ssh_use_tunnel:
|
if revpi_settings.ssh_use_tunnel:
|
||||||
# todo: Change message, that user can use ssh
|
self.connect_error.emit(
|
||||||
QtWidgets.QMessageBox.critical(
|
self.tr("Error"), self.tr(
|
||||||
parent, self.tr("Error"), self.tr(
|
"Can not connect to RevPiPyLoad service through SSH tunnel!\n\n"
|
||||||
"Can not connect to RevPi XML-RPC Service! \n\n"
|
"This could have the following reasons:\n"
|
||||||
"This could have the following reasons: The RevPi is not "
|
"- The RevPiPyLoad service is not running (activate it on your Revolution Pi)\n"
|
||||||
"online, the XML-RPC service is not running / bind to "
|
"- The RevPiPyLoad XML-RPC service is NOT bind to localhost\n"
|
||||||
"localhost or the ACL permission is not set for your "
|
"- The ACL permission is not set for 127.0.0.1!!!"
|
||||||
"IP!!!\n\nRun 'sudo revpipyload_secure_installation' on "
|
),
|
||||||
"Revolution Pi to setup this function!"
|
ConnectionFail.NO_XML_RPC_VIA_TUNNEL,
|
||||||
|
revpi_settings,
|
||||||
)
|
)
|
||||||
|
else:
|
||||||
|
self.connect_error.emit(
|
||||||
|
self.tr("Error"), self.tr(
|
||||||
|
"Can not connect to RevPiPyLoad XML-RPC service! \n\n"
|
||||||
|
"This could have the following reasons:\n"
|
||||||
|
"- The Revolution Pi is not online\n"
|
||||||
|
"- The RevPiPyLoad service is not running (activate it on your Revolution Pi)\n"
|
||||||
|
"- The RevPiPyLoad XML-RPC service is bind to localhost, only\n"
|
||||||
|
"- The ACL permission is not set for your IP!!!\n\n"
|
||||||
|
"Use 'Connect via SSH' to use an encrypted connection or run "
|
||||||
|
"'sudo revpipyload_secure_installation' on Revolution Pi to setup direct remote access!"
|
||||||
|
),
|
||||||
|
ConnectionFail.NO_XML_RPC,
|
||||||
|
revpi_settings,
|
||||||
)
|
)
|
||||||
|
|
||||||
return False
|
return False
|
||||||
@@ -404,11 +423,11 @@ class ConnectionManager(QtCore.QThread):
|
|||||||
self.settings = revpi_settings
|
self.settings = revpi_settings
|
||||||
self.ssh_pass = ssh_pass
|
self.ssh_pass = ssh_pass
|
||||||
self.pyload_version = pyload_version
|
self.pyload_version = pyload_version
|
||||||
|
self.pyload_version_str = pyload_version_str
|
||||||
self.xml_funcs = xml_funcs
|
self.xml_funcs = xml_funcs
|
||||||
self.xml_mode = xml_mode
|
self.xml_mode = xml_mode
|
||||||
|
|
||||||
with self._lck_cli:
|
with self._lck_cli:
|
||||||
socket.setdefaulttimeout(revpi_settings.timeout)
|
|
||||||
self.ssh_tunnel_server = ssh_tunnel_server
|
self.ssh_tunnel_server = ssh_tunnel_server
|
||||||
self._cli = sp
|
self._cli = sp
|
||||||
self._cli_connect.put_nowait((
|
self._cli_connect.put_nowait((
|
||||||
@@ -457,7 +476,13 @@ class ConnectionManager(QtCore.QThread):
|
|||||||
self.connection_disconnected.emit()
|
self.connection_disconnected.emit()
|
||||||
|
|
||||||
def pyload_simulate(self, configrsc: str, procimg: str, clean_existing: bool):
|
def pyload_simulate(self, configrsc: str, procimg: str, clean_existing: bool):
|
||||||
"""Start the simulator for piControl on local computer."""
|
"""
|
||||||
|
Start the simulator for piControl on local computer.
|
||||||
|
|
||||||
|
:param configrsc: piCtory configuration
|
||||||
|
:param procimg: Process image, which is a 4 kByte file for simulation
|
||||||
|
:param clean_existing: Reset the file to ZERO \x00 bytes
|
||||||
|
"""
|
||||||
pi.logger.debug("ConnectionManager.start_simulate")
|
pi.logger.debug("ConnectionManager.start_simulate")
|
||||||
|
|
||||||
if not exists(procimg) or clean_existing:
|
if not exists(procimg) or clean_existing:
|
||||||
@@ -671,6 +696,16 @@ def all_revpi_settings() -> [RevPiSettings]:
|
|||||||
return [RevPiSettings(i) for i in range(count_settings)]
|
return [RevPiSettings(i) for i in range(count_settings)]
|
||||||
|
|
||||||
|
|
||||||
|
def swap_word_order(bytes_to_swap) -> bytes:
|
||||||
|
"""Swap word order of an even byte array."""
|
||||||
|
array_lenght = len(bytes_to_swap)
|
||||||
|
swap_array = bytearray(bytes_to_swap)
|
||||||
|
for i in range(0, array_lenght // 2, 2):
|
||||||
|
swap_array[-i - 2:array_lenght - i], swap_array[i:i + 2] = \
|
||||||
|
swap_array[i:i + 2], swap_array[-i - 2:array_lenght - i]
|
||||||
|
return bytes(swap_array)
|
||||||
|
|
||||||
|
|
||||||
def import_old_settings():
|
def import_old_settings():
|
||||||
"""Try to import saved connections from old storage to new setting object."""
|
"""Try to import saved connections from old storage to new setting object."""
|
||||||
if settings.value("revpicommander/imported_settings", False, type=bool):
|
if settings.value("revpicommander/imported_settings", False, type=bool):
|
||||||
@@ -686,7 +721,8 @@ def import_old_settings():
|
|||||||
revpi_setting = RevPiSettings(i, settings_storage=old_settings)
|
revpi_setting = RevPiSettings(i, settings_storage=old_settings)
|
||||||
revpi_setting._settings = settings
|
revpi_setting._settings = settings
|
||||||
revpi_setting.save_settings()
|
revpi_setting.save_settings()
|
||||||
except Exception as e:
|
except Exception:
|
||||||
pi.logger.warning("Could not import saved connection {0}".format(i))
|
pi.logger.warning("Could not import saved connection {0}".format(i))
|
||||||
|
|
||||||
|
|
||||||
import_old_settings()
|
import_old_settings()
|
||||||
|
|||||||
Binary file not shown.
@@ -93,70 +93,198 @@ Nicht gespeicherte Änderunen gehen verloren</translation>
|
|||||||
<translation type="obsolete">Der ausgewählte RevPi ist schon in der Verbindungsliste als '{0}'.</translation>
|
<translation type="obsolete">Der ausgewählte RevPi ist schon in der Verbindungsliste als '{0}'.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../avahisearch.py" line="283"/>
|
<location filename="../avahisearch.py" line="288"/>
|
||||||
<source> over SSH</source>
|
<source> over SSH</source>
|
||||||
<translation> über SSH</translation>
|
<translation> über SSH</translation>
|
||||||
</message>
|
</message>
|
||||||
</context>
|
</context>
|
||||||
|
<context>
|
||||||
|
<name>ConnectingPyload</name>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="299"/>
|
||||||
|
<source>Simulator started...</source>
|
||||||
|
<translation type="obsolete">Simulator gestartet...</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="299"/>
|
||||||
|
<source>The simulator is running!
|
||||||
|
|
||||||
|
You can work with this simulator if your call RevPiModIO with this additional parameters:
|
||||||
|
procimg={0}
|
||||||
|
configrsc={1}
|
||||||
|
|
||||||
|
You can copy that from header textbox.</source>
|
||||||
|
<translation type="obsolete">Der Simulator läuft!
|
||||||
|
|
||||||
|
Du kannst mit der Simulation arbeiten, wenn du RevPiModIO mit diesen zusätzlichen Parametern instantiierst:
|
||||||
|
procimg={0}
|
||||||
|
configrsc={1}
|
||||||
|
|
||||||
|
Dies kann aus der Textbox oben kopiert werden.</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="308"/>
|
||||||
|
<source>Can not start...</source>
|
||||||
|
<translation type="obsolete">Kann nicht gestartet werden...</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="308"/>
|
||||||
|
<source>Can not start the simulator! Maybe the piCtory file is corrupt or you have no write permissions for '{0}'.</source>
|
||||||
|
<translation type="obsolete">Kann Simulator nicht starten! Vielleicht ist die piCtory Datei defekt oder es gibt keine Schreibberechtigung für '{0}'.</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="501"/>
|
||||||
|
<source>Warning</source>
|
||||||
|
<translation type="obsolete">Warnung</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="324"/>
|
||||||
|
<source>This version of Logviewer ist not supported in version {0} of RevPiPyLoad on your RevPi! You need at least version 0.4.1.</source>
|
||||||
|
<translation type="obsolete">Diese Version vom Logbetrachter wird in RevPiPyLoad Version {0} nicht unterstützt! Es wird mindestens Version 0.4.1 benötigt.</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="374"/>
|
||||||
|
<source>XML-RPC access mode in the RevPiPyLoad configuration is too small to access this dialog!</source>
|
||||||
|
<translation type="obsolete">XML-RPC Zugriffsberechtigung in der RevPiPyLoad Konfiguraiton ist zu klein für diese Einstellungen!</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="528"/>
|
||||||
|
<source>Error</source>
|
||||||
|
<translation type="obsolete">Fehler</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="354"/>
|
||||||
|
<source>The Version of RevPiPyLoad on your Revolution Pi ({0}) is to old. This Version of RevPiCommander require at least version 0.6.0 of RevPiPyLoad. Please update your Revolution Pi!</source>
|
||||||
|
<translation type="obsolete">Die Version von RevPiPyLoad ({0}) auf dem Revolution Pi ist zu alt. Diese Version vom RevPiCommander braucht mindestens Version 0.6.0. Bitte aktualisiere deinen Revolution Pi!</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="407"/>
|
||||||
|
<source>Question</source>
|
||||||
|
<translation type="obsolete">Frage</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="407"/>
|
||||||
|
<source>Are you sure to reset piControl?
|
||||||
|
The pictory configuration will be reloaded. During that time the process image will be interrupted and could rise errors on running control programs!</source>
|
||||||
|
<translation type="obsolete">Soll piControl wirklich zurückgesetzt werden?
|
||||||
|
Die piCtory Konfiguration wird neu geladen. Das Prozessabbild wird in dieser Zeit nicht verfügbar sein und es könnten Fehler in Steuerungsprogrammen ausgelöst werden!</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="420"/>
|
||||||
|
<source>Success</source>
|
||||||
|
<translation type="obsolete">Erfolgreich</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="420"/>
|
||||||
|
<source>piControl reset executed successfully</source>
|
||||||
|
<translation type="obsolete">piControl wurde erfolgreich zurückgesetzt</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="427"/>
|
||||||
|
<source>piControl reset could not be executed successfully</source>
|
||||||
|
<translation type="obsolete">piControl konnte nicht zurückgesetzt werden</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="475"/>
|
||||||
|
<source>Reset to piCtory defaults...</source>
|
||||||
|
<translation type="obsolete">Standardwerte von piCtory laden...</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="475"/>
|
||||||
|
<source>Do you want to reset your process image to {0} values?
|
||||||
|
You have to stop other RevPiModIO programs before doing that, because they could reset the outputs.</source>
|
||||||
|
<translation type="obsolete">Soll das virtuelle Prozessabbild auf {0} zurückgesetzt werden?
|
||||||
|
Es sollten alle RevPiModIO Programme vorher beendet werden, da diese ihre IO Werte sofort wieder schreiben würden.</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="475"/>
|
||||||
|
<source>zero</source>
|
||||||
|
<translation type="obsolete">null</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="475"/>
|
||||||
|
<source>piCtory default</source>
|
||||||
|
<translation type="obsolete">piCtory Standardwerte</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="501"/>
|
||||||
|
<source>The watch mode ist not supported in version {0} of RevPiPyLoad on your RevPi! You need at least version 0.5.3! Maybe the python3-revpimodio2 module is not installed on your RevPi at least version 2.0.0.</source>
|
||||||
|
<translation type="obsolete">Der SPS Betrachter ist in Version {0} von RevPiPyLoad auf dem Rev Pi nicht unterstützt! Es muss mindestens Version 0.5.3 installiert sein! Vielleicht fehlt auch das python3-revpimodio2 Modul, welches mindestens Version 2.0.0 haben muss.</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="513"/>
|
||||||
|
<source>Can not load this function, because your ACL level is to low!
|
||||||
|
You need at least level 1 to read or level 3 to write.</source>
|
||||||
|
<translation type="obsolete">Für diese Funktion ist das Berechtigungslevel zu gering!
|
||||||
|
Es muss mindestens Level 1 zum Lesen oder Level 3 zu Schreiben sein.</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="528"/>
|
||||||
|
<source>Can not load piCtory configuration.
|
||||||
|
Did you create a hardware configuration? Please check this in piCtory!</source>
|
||||||
|
<translation type="obsolete">Kann piCtory Konfiguration nicht laden.
|
||||||
|
Wurde eine Hardwarekonfiguration in piCtory erzeugt?</translation>
|
||||||
|
</message>
|
||||||
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>ConnectionManager</name>
|
<name>ConnectionManager</name>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="512"/>
|
<location filename="../helper.py" line="538"/>
|
||||||
<source>SIMULATING</source>
|
<source>SIMULATING</source>
|
||||||
<translation>SIMULATION</translation>
|
<translation>SIMULATION</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="515"/>
|
<location filename="../helper.py" line="541"/>
|
||||||
<source>NOT CONNECTED</source>
|
<source>NOT CONNECTED</source>
|
||||||
<translation>NICHT VERBUNDEN</translation>
|
<translation>NICHT VERBUNDEN</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="532"/>
|
<location filename="../helper.py" line="558"/>
|
||||||
<source>SERVER ERROR</source>
|
<source>SERVER ERROR</source>
|
||||||
<translation>SERVER FEHLER</translation>
|
<translation>SERVER FEHLER</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="557"/>
|
<location filename="../helper.py" line="583"/>
|
||||||
<source>RUNNING</source>
|
<source>RUNNING</source>
|
||||||
<translation>LÄUFT</translation>
|
<translation>LÄUFT</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="559"/>
|
<location filename="../helper.py" line="585"/>
|
||||||
<source>PLC FILE NOT FOUND</source>
|
<source>PLC FILE NOT FOUND</source>
|
||||||
<translation>SPS PROGRAMM NICHT GEFUNDEN</translation>
|
<translation>SPS PROGRAMM NICHT GEFUNDEN</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="561"/>
|
<location filename="../helper.py" line="587"/>
|
||||||
<source>NOT RUNNING (NO STATUS)</source>
|
<source>NOT RUNNING (NO STATUS)</source>
|
||||||
<translation>LÄUFT NICHT (KEIN STATUS)</translation>
|
<translation>LÄUFT NICHT (KEIN STATUS)</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="563"/>
|
<location filename="../helper.py" line="589"/>
|
||||||
<source>PROGRAM KILLED</source>
|
<source>PROGRAM KILLED</source>
|
||||||
<translation>PROGRAMM GETÖTET</translation>
|
<translation>PROGRAMM GETÖTET</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="565"/>
|
<location filename="../helper.py" line="591"/>
|
||||||
<source>PROGRAM TERMED</source>
|
<source>PROGRAM TERMED</source>
|
||||||
<translation>PROGRAMM BEENDET</translation>
|
<translation>PROGRAMM BEENDET</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="567"/>
|
<location filename="../helper.py" line="593"/>
|
||||||
<source>NOT RUNNING</source>
|
<source>NOT RUNNING</source>
|
||||||
<translation>LÄUFT NICHT</translation>
|
<translation>LÄUFT NICHT</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="569"/>
|
<location filename="../helper.py" line="595"/>
|
||||||
<source>FINISHED WITH CODE {0}</source>
|
<source>FINISHED WITH CODE {0}</source>
|
||||||
<translation>BEENDET MIT CODE {0}</translation>
|
<translation>BEENDET MIT CODE {0}</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="384"/>
|
<location filename="../helper.py" line="406"/>
|
||||||
<source>Error</source>
|
<source>Error</source>
|
||||||
<translation>Fehler</translation>
|
<translation>Fehler</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="352"/>
|
<location filename="../helper.py" line="356"/>
|
||||||
<source>The combination of username and password was rejected from the SSH server.
|
<source>The combination of username and password was rejected from the SSH server.
|
||||||
|
|
||||||
Try again.</source>
|
Try again.</source>
|
||||||
@@ -165,7 +293,7 @@ Try again.</source>
|
|||||||
Bitte erneut versuchen.</translation>
|
Bitte erneut versuchen.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="361"/>
|
<location filename="../helper.py" line="368"/>
|
||||||
<source>Could not establish a SSH connection to server:
|
<source>Could not establish a SSH connection to server:
|
||||||
|
|
||||||
{0}</source>
|
{0}</source>
|
||||||
@@ -174,18 +302,91 @@ Bitte erneut versuchen.</translation>
|
|||||||
{0}</translation>
|
{0}</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../helper.py" line="384"/>
|
<location filename="../helper.py" line="395"/>
|
||||||
<source>Can not connect to RevPi XML-RPC Service!
|
<source>Can not connect to RevPi XML-RPC Service!
|
||||||
|
|
||||||
This could have the following reasons: The RevPi is not online, the XML-RPC service is not running / bind to localhost or the ACL permission is not set for your IP!!!
|
This could have the following reasons: The RevPi is not online, the XML-RPC service is not running / bind to localhost or the ACL permission is not set for your IP!!!
|
||||||
|
|
||||||
Run 'sudo revpipyload_secure_installation' on Revolution Pi to setup this function!</source>
|
Run 'sudo revpipyload_secure_installation' on Revolution Pi to setup this function!</source>
|
||||||
<translation>Kann keine Verbindung zum RevPi XML-RPC Dienst herstellen!
|
<translation type="obsolete">Kann keine Verbindung zum RevPi XML-RPC Dienst herstellen!
|
||||||
|
|
||||||
Das kann eine der folgenden Ursachen haben: Der Rev Pi ist nicht online, der XML-RPC Dienst läuft nicht / ist an localhost gebunden order die Berechtigungen sind nicht für diese IP gesetzt!!!
|
Das kann eine der folgenden Ursachen haben: Der Rev Pi ist nicht online, der XML-RPC Dienst läuft nicht / ist an localhost gebunden order die Berechtigungen sind nicht für diese IP gesetzt!!!
|
||||||
|
|
||||||
Führe 'sudo revpipyload_secure_installation' auf dem Revolution Pi aus um diese Funktion zu konfigurieren!</translation>
|
Führe 'sudo revpipyload_secure_installation' auf dem Revolution Pi aus um diese Funktion zu konfigurieren!</translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../helper.py" line="383"/>
|
||||||
|
<source>Can not connect to RevPi XML-RPC Service through SSH tunnel!
|
||||||
|
|
||||||
|
This could have the following reasons: The XML-RPC service is not running / bind to localhost or the ACL permission is not set for 127.0.0.1!!!</source>
|
||||||
|
<translation type="obsolete">Kann keine Verbindung zum RevPi XML-RPC Dienst herstellen!
|
||||||
|
|
||||||
|
Das kann eine der folgenden Ursachen haben: Der XML-RPC Dienst läuft nicht / ist nicht an localhost gebunden order die Berechtigungen sind nicht für 127.0.0.1 gesetzt!!!</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../helper.py" line="383"/>
|
||||||
|
<source>Can not connect to RevPi XML-RPC Service through SSH tunnel!
|
||||||
|
|
||||||
|
This could have the following reasons: The XML-RPC service is not running / not bind to localhost or the ACL permission is not set for 127.0.0.1!!!</source>
|
||||||
|
<translation type="obsolete">Kann keine Verbindung zum RevPi XML-RPC Dienst über SSH herstellen!
|
||||||
|
|
||||||
|
Das kann eine der folgenden Ursachen haben: Der XML-RPC Dienst läuft nicht / ist nicht an localhost gebunden order die Berechtigungen sind nicht für 127.0.0.1 gesetzt!!!</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../helper.py" line="389"/>
|
||||||
|
<source>Can not connect to RevPiPyLoad service through SSH tunnel!
|
||||||
|
|
||||||
|
This could have the following reasons:
|
||||||
|
- The RevPiPyLoad service is not running (activate it on your Revolution Pi)
|
||||||
|
- The RevPiPyLoad XML-RPC service is NOT bind to localhost
|
||||||
|
- The ACL permission is not set for 127.0.0.1!!!
|
||||||
|
|
||||||
|
</source>
|
||||||
|
<translation type="obsolete">Kann keine Verbindung zum RevPiPyLoad Dienst über SSH herstellen!
|
||||||
|
|
||||||
|
Das kann eine der folgenden Ursachen haben:
|
||||||
|
- Der RevPiPyLoad Dienst läuft nicht (aktiviere Diesen auf dem Revolution Pi)
|
||||||
|
- Der RevPiPyLoad XML-RPC Dienst ist NICHT an localhost gebunden
|
||||||
|
- Die Berechtigungen sind nicht für 127.0.0.1 gesetzt!!!
|
||||||
|
|
||||||
|
Benutze "Über SSH verbinden" um eine verschlüsselte Verbindung aufzubauen oder führe 'sudo revpipyload_secure_installation' auf dem Revolution Pi aus, um eine direkte Verbindung zu konfigurieren!</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../helper.py" line="406"/>
|
||||||
|
<source>Can not connect to RevPiPyLoad XML-RPC service!
|
||||||
|
|
||||||
|
This could have the following reasons:
|
||||||
|
- The Revolution Pi is not online
|
||||||
|
- The RevPiPyLoad service is not running (activate it on your Revolution Pi)
|
||||||
|
- The RevPiPyLoad XML-RPC service is bind to localhost, only
|
||||||
|
- The ACL permission is not set for your IP!!!
|
||||||
|
|
||||||
|
Use 'Connect via SSH' to use an encrypted connection or run 'sudo revpipyload_secure_installation' on Revolution Pi to setup direct remote access!</source>
|
||||||
|
<translation>Kann keine Verbindung zum RevPiPyLoad XML-RPC Dienst herstellen!
|
||||||
|
|
||||||
|
Das kann eine der folgenden Ursachen haben:
|
||||||
|
- Der Revolution Pi ist nicht online
|
||||||
|
- Der RevPiPyLoad Dienst läuft nicht (aktiviere Diesen auf dem Revolution Pi)
|
||||||
|
- Der RevPiPyLoad XML-RPC Dienst ist nur an localhost gebunden
|
||||||
|
- Die Berechtigungen sind nicht für diese IP gesetzt!!!
|
||||||
|
|
||||||
|
Benutze "Über SSH verbinden" um eine verschlüsselte Verbindung aufzubauen oder führe 'sudo revpipyload_secure_installation' auf dem Revolution Pi aus, um eine direkte Verbindung zu konfigurieren!</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../helper.py" line="394"/>
|
||||||
|
<source>Can not connect to RevPiPyLoad service through SSH tunnel!
|
||||||
|
|
||||||
|
This could have the following reasons:
|
||||||
|
- The RevPiPyLoad service is not running (activate it on your Revolution Pi)
|
||||||
|
- The RevPiPyLoad XML-RPC service is NOT bind to localhost
|
||||||
|
- The ACL permission is not set for 127.0.0.1!!!</source>
|
||||||
|
<translation>Kann keine Verbindung zum RevPiPyLoad Dienst über SSH herstellen!
|
||||||
|
|
||||||
|
Das kann eine der folgenden Ursachen haben:
|
||||||
|
- Der RevPiPyLoad Dienst läuft nicht (aktiviere Diesen auf dem Revolution Pi)
|
||||||
|
- Der RevPiPyLoad XML-RPC Dienst ist NICHT an localhost gebunden
|
||||||
|
- Die Berechtigungen sind nicht für 127.0.0.1 gesetzt!!!</translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>DebugControl</name>
|
<name>DebugControl</name>
|
||||||
@@ -200,24 +401,24 @@ Führe 'sudo revpipyload_secure_installation' auf dem Revolution Pi au
|
|||||||
<translation>Fehler bei Werteempfang von RevPi.</translation>
|
<translation>Fehler bei Werteempfang von RevPi.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../debugcontrol.py" line="228"/>
|
<location filename="../debugcontrol.py" line="229"/>
|
||||||
<source>Auto update values...</source>
|
<source>Auto update values...</source>
|
||||||
<translation>Werte automatisch aktualisiert...</translation>
|
<translation>Werte automatisch aktualisiert...</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../debugcontrol.py" line="230"/>
|
<location filename="../debugcontrol.py" line="231"/>
|
||||||
<source>Values updated...</source>
|
<source>Values updated...</source>
|
||||||
<translation>Werte aktualisiert...</translation>
|
<translation>Werte aktualisiert...</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../debugcontrol.py" line="267"/>
|
<location filename="../debugcontrol.py" line="268"/>
|
||||||
<source>Error set value of device '{0}' Output '{1}': {2}
|
<source>Error set value of device '{0}' Output '{1}': {2}
|
||||||
</source>
|
</source>
|
||||||
<translation>Fehler beim Setzen des Ausgangs '{1}' auf Modul '{0}': {2}
|
<translation>Fehler beim Setzen des Ausgangs '{1}' auf Modul '{0}': {2}
|
||||||
</translation>
|
</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../debugcontrol.py" line="276"/>
|
<location filename="../debugcontrol.py" line="277"/>
|
||||||
<source>Error</source>
|
<source>Error</source>
|
||||||
<translation>Fehler</translation>
|
<translation>Fehler</translation>
|
||||||
</message>
|
</message>
|
||||||
@@ -225,35 +426,40 @@ Führe 'sudo revpipyload_secure_installation' auf dem Revolution Pi au
|
|||||||
<context>
|
<context>
|
||||||
<name>DebugIos</name>
|
<name>DebugIos</name>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../debugios.py" line="207"/>
|
<location filename="../debugios.py" line="228"/>
|
||||||
<source>signed</source>
|
<source>signed</source>
|
||||||
<translation></translation>
|
<translation></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../debugios.py" line="212"/>
|
<location filename="../debugios.py" line="233"/>
|
||||||
<source>big_endian</source>
|
<source>big_endian</source>
|
||||||
<translation></translation>
|
<translation></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../debugios.py" line="201"/>
|
<location filename="../debugios.py" line="219"/>
|
||||||
<source>as text</source>
|
<source>as text</source>
|
||||||
<translation></translation>
|
<translation></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../debugios.py" line="203"/>
|
<location filename="../debugios.py" line="221"/>
|
||||||
<source>as number</source>
|
<source>as number</source>
|
||||||
<translation></translation>
|
<translation></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../debugios.py" line="346"/>
|
<location filename="../debugios.py" line="377"/>
|
||||||
<source>Can not use format text</source>
|
<source>Can not use format text</source>
|
||||||
<translation>Formatierung nicht möglich</translation>
|
<translation>Formatierung nicht möglich</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../debugios.py" line="346"/>
|
<location filename="../debugios.py" line="377"/>
|
||||||
<source>Can not convert bytes {0} to a text for IO '{1}'. Switch to number format instead!</source>
|
<source>Can not convert bytes {0} to a text for IO '{1}'. Switch to number format instead!</source>
|
||||||
<translation>Kann bytes {0} für '{1}' nicht in Text konvertieren. Wechseln Sie auf Nummernformat!</translation>
|
<translation>Kann bytes {0} für '{1}' nicht in Text konvertieren. Wechseln Sie auf Nummernformat!</translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../debugios.py" line="239"/>
|
||||||
|
<source>switch wordorder</source>
|
||||||
|
<translation>Wordorder tauschen</translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>MqttManager</name>
|
<name>MqttManager</name>
|
||||||
@@ -283,90 +489,90 @@ Ungesicherte Änderungen gehen verloren.</translation>
|
|||||||
<context>
|
<context>
|
||||||
<name>RevPiCommander</name>
|
<name>RevPiCommander</name>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="222"/>
|
<location filename="../revpicommander.py" line="320"/>
|
||||||
<source>Simulator started...</source>
|
<source>Simulator started...</source>
|
||||||
<translation>Simulator gestartet...</translation>
|
<translation>Simulator gestartet...</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="231"/>
|
<location filename="../revpicommander.py" line="329"/>
|
||||||
<source>Can not start...</source>
|
<source>Can not start...</source>
|
||||||
<translation>Kann nicht gestartet werden...</translation>
|
<translation>Kann nicht gestartet werden...</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="424"/>
|
<location filename="../revpicommander.py" line="522"/>
|
||||||
<source>Warning</source>
|
<source>Warning</source>
|
||||||
<translation>Warnung</translation>
|
<translation>Warnung</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="247"/>
|
<location filename="../revpicommander.py" line="345"/>
|
||||||
<source>This version of Logviewer ist not supported in version {0} of RevPiPyLoad on your RevPi! You need at least version 0.4.1.</source>
|
<source>This version of Logviewer ist not supported in version {0} of RevPiPyLoad on your RevPi! You need at least version 0.4.1.</source>
|
||||||
<translation>Diese Version vom Logbetrachter wird in RevPiPyLoad Version {0} nicht unterstützt! Es wird mindestens Version 0.4.1 benötigt.</translation>
|
<translation>Diese Version vom Logbetrachter wird in RevPiPyLoad Version {0} nicht unterstützt! Es wird mindestens Version 0.4.1 benötigt.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="297"/>
|
<location filename="../revpicommander.py" line="395"/>
|
||||||
<source>XML-RPC access mode in the RevPiPyLoad configuration is too small to access this dialog!</source>
|
<source>XML-RPC access mode in the RevPiPyLoad configuration is too small to access this dialog!</source>
|
||||||
<translation>XML-RPC Zugriffsberechtigung in der RevPiPyLoad Konfiguraiton ist zu klein für diese Einstellungen!</translation>
|
<translation>XML-RPC Zugriffsberechtigung in der RevPiPyLoad Konfiguraiton ist zu klein für diese Einstellungen!</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="451"/>
|
<location filename="../revpicommander.py" line="549"/>
|
||||||
<source>Error</source>
|
<source>Error</source>
|
||||||
<translation>Fehler</translation>
|
<translation>Fehler</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="277"/>
|
<location filename="../revpicommander.py" line="375"/>
|
||||||
<source>The Version of RevPiPyLoad on your Revolution Pi ({0}) is to old. This Version of RevPiCommander require at least version 0.6.0 of RevPiPyLoad. Please update your Revolution Pi!</source>
|
<source>The Version of RevPiPyLoad on your Revolution Pi ({0}) is to old. This Version of RevPiCommander require at least version 0.6.0 of RevPiPyLoad. Please update your Revolution Pi!</source>
|
||||||
<translation>Die Version von RevPiPyLoad ({0}) auf dem Revolution Pi ist zu alt. Diese Version vom RevPiCommander braucht mindestens Version 0.6.0. Bitte aktualisiere deinen Revolution Pi!</translation>
|
<translation>Die Version von RevPiPyLoad ({0}) auf dem Revolution Pi ist zu alt. Diese Version vom RevPiCommander braucht mindestens Version 0.6.0. Bitte aktualisiere deinen Revolution Pi!</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="330"/>
|
<location filename="../revpicommander.py" line="428"/>
|
||||||
<source>Question</source>
|
<source>Question</source>
|
||||||
<translation>Frage</translation>
|
<translation>Frage</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="330"/>
|
<location filename="../revpicommander.py" line="428"/>
|
||||||
<source>Are you sure to reset piControl?
|
<source>Are you sure to reset piControl?
|
||||||
The pictory configuration will be reloaded. During that time the process image will be interrupted and could rise errors on running control programs!</source>
|
The pictory configuration will be reloaded. During that time the process image will be interrupted and could rise errors on running control programs!</source>
|
||||||
<translation>Soll piControl wirklich zurückgesetzt werden?
|
<translation>Soll piControl wirklich zurückgesetzt werden?
|
||||||
Die piCtory Konfiguration wird neu geladen. Das Prozessabbild wird in dieser Zeit nicht verfügbar sein und es könnten Fehler in Steuerungsprogrammen ausgelöst werden!</translation>
|
Die piCtory Konfiguration wird neu geladen. Das Prozessabbild wird in dieser Zeit nicht verfügbar sein und es könnten Fehler in Steuerungsprogrammen ausgelöst werden!</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="343"/>
|
<location filename="../revpicommander.py" line="441"/>
|
||||||
<source>Success</source>
|
<source>Success</source>
|
||||||
<translation>Erfolgreich</translation>
|
<translation>Erfolgreich</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="343"/>
|
<location filename="../revpicommander.py" line="441"/>
|
||||||
<source>piControl reset executed successfully</source>
|
<source>piControl reset executed successfully</source>
|
||||||
<translation>piControl wurde erfolgreich zurückgesetzt</translation>
|
<translation>piControl wurde erfolgreich zurückgesetzt</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="350"/>
|
<location filename="../revpicommander.py" line="448"/>
|
||||||
<source>piControl reset could not be executed successfully</source>
|
<source>piControl reset could not be executed successfully</source>
|
||||||
<translation>piControl konnte nicht zurückgesetzt werden</translation>
|
<translation>piControl konnte nicht zurückgesetzt werden</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="398"/>
|
<location filename="../revpicommander.py" line="496"/>
|
||||||
<source>Reset to piCtory defaults...</source>
|
<source>Reset to piCtory defaults...</source>
|
||||||
<translation>Standardwerte von piCtory laden...</translation>
|
<translation>Standardwerte von piCtory laden...</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="424"/>
|
<location filename="../revpicommander.py" line="522"/>
|
||||||
<source>The watch mode ist not supported in version {0} of RevPiPyLoad on your RevPi! You need at least version 0.5.3! Maybe the python3-revpimodio2 module is not installed on your RevPi at least version 2.0.0.</source>
|
<source>The watch mode ist not supported in version {0} of RevPiPyLoad on your RevPi! You need at least version 0.5.3! Maybe the python3-revpimodio2 module is not installed on your RevPi at least version 2.0.0.</source>
|
||||||
<translation>Der SPS Betrachter ist in Version {0} von RevPiPyLoad auf dem Rev Pi nicht unterstützt! Es muss mindestens Version 0.5.3 installiert sein! Vielleicht fehlt auch das python3-revpimodio2 Modul, welches mindestens Version 2.0.0 haben muss.</translation>
|
<translation>Der SPS Betrachter ist in Version {0} von RevPiPyLoad auf dem Rev Pi nicht unterstützt! Es muss mindestens Version 0.5.3 installiert sein! Vielleicht fehlt auch das python3-revpimodio2 Modul, welches mindestens Version 2.0.0 haben muss.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="436"/>
|
<location filename="../revpicommander.py" line="534"/>
|
||||||
<source>Can not load this function, because your ACL level is to low!
|
<source>Can not load this function, because your ACL level is to low!
|
||||||
You need at least level 1 to read or level 3 to write.</source>
|
You need at least level 1 to read or level 3 to write.</source>
|
||||||
<translation>Für diese Funktion ist das Berechtigungslevel zu gering!
|
<translation>Für diese Funktion ist das Berechtigungslevel zu gering!
|
||||||
Es muss mindestens Level 1 zum Lesen oder Level 3 zu Schreiben sein.</translation>
|
Es muss mindestens Level 1 zum Lesen oder Level 3 zu Schreiben sein.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="451"/>
|
<location filename="../revpicommander.py" line="549"/>
|
||||||
<source>Can not load piCtory configuration.
|
<source>Can not load piCtory configuration.
|
||||||
Did you create a hardware configuration? Please check this in piCtory!</source>
|
Did you create a hardware configuration? Please check this in piCtory!</source>
|
||||||
<translation>Kann piCtory Konfiguration nicht laden.
|
<translation>Kann piCtory Konfiguration nicht laden.
|
||||||
Wurde eine Hardwarekonfiguration in piCtory erzeugt?</translation>
|
Wurde eine Hardwarekonfiguration in piCtory erzeugt? Bitte prüfe dies in piCtory!</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="101"/>
|
<location filename="../revpicommander.py" line="101"/>
|
||||||
@@ -382,7 +588,7 @@ Das kann eine der folgenden Ursachen haben: Der Rev Pi ist nicht online, der XML
|
|||||||
Führe 'sudo revpipyload_secure_installation' auf dem Revolution Pi aus um diese Funktion zu konfigurieren!</translation>
|
Führe 'sudo revpipyload_secure_installation' auf dem Revolution Pi aus um diese Funktion zu konfigurieren!</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="222"/>
|
<location filename="../revpicommander.py" line="320"/>
|
||||||
<source>The simulator is running!
|
<source>The simulator is running!
|
||||||
|
|
||||||
You can work with this simulator if your call RevPiModIO with this additional parameters:
|
You can work with this simulator if your call RevPiModIO with this additional parameters:
|
||||||
@@ -399,27 +605,65 @@ configrsc={1}
|
|||||||
Dies kann aus der Textbox oben kopiert werden.</translation>
|
Dies kann aus der Textbox oben kopiert werden.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="231"/>
|
<location filename="../revpicommander.py" line="329"/>
|
||||||
<source>Can not start the simulator! Maybe the piCtory file is corrupt or you have no write permissions for '{0}'.</source>
|
<source>Can not start the simulator! Maybe the piCtory file is corrupt or you have no write permissions for '{0}'.</source>
|
||||||
<translation>Kann Simulator nicht starten! Vielleicht ist die piCtory Datei defekt oder es gibt keine Schreibberechtigung für '{0}'.</translation>
|
<translation>Kann Simulator nicht starten! Vielleicht ist die piCtory Datei defekt oder es gibt keine Schreibberechtigung für '{0}'.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="398"/>
|
<location filename="../revpicommander.py" line="496"/>
|
||||||
<source>Do you want to reset your process image to {0} values?
|
<source>Do you want to reset your process image to {0} values?
|
||||||
You have to stop other RevPiModIO programs before doing that, because they could reset the outputs.</source>
|
You have to stop other RevPiModIO programs before doing that, because they could reset the outputs.</source>
|
||||||
<translation>Soll das virtuelle Prozessabbild auf {0} zurückgesetzt werden?
|
<translation>Soll das virtuelle Prozessabbild auf {0} zurückgesetzt werden?
|
||||||
Es sollten alle RevPiModIO Programme vorher beendet werden, da diese ihre IO Werte sofort wieder schreiben würden.</translation>
|
Es sollten alle RevPiModIO Programme vorher beendet werden, da diese ihre IO Werte sofort wieder schreiben würden.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="398"/>
|
<location filename="../revpicommander.py" line="496"/>
|
||||||
<source>zero</source>
|
<source>zero</source>
|
||||||
<translation>null</translation>
|
<translation>null</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpicommander.py" line="398"/>
|
<location filename="../revpicommander.py" line="496"/>
|
||||||
<source>piCtory default</source>
|
<source>piCtory default</source>
|
||||||
<translation>piCtory Standardwerte</translation>
|
<translation>piCtory Standardwerte</translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="250"/>
|
||||||
|
<source>Revolution Pi connected!</source>
|
||||||
|
<translation>Revolution Pi verbunden!</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="287"/>
|
||||||
|
<source>Connecting...</source>
|
||||||
|
<translation>Verbinde...</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="250"/>
|
||||||
|
<source>Establish a connection to the Revolution Pi...</source>
|
||||||
|
<translation>Baue eine Verbindung zum Revolution Pi auf...</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="133"/>
|
||||||
|
<source>Can not connect to RevPiPyLoad service through SSH tunnel!
|
||||||
|
|
||||||
|
We are trying to activate this service now and reconnect. The settings can be changed at any time via web status.</source>
|
||||||
|
<translation type="obsolete">Über den SSH Tunnel kann keine Verbindung zu RevPiPyLoad hergestellt werden!
|
||||||
|
|
||||||
|
Wir werden versuchen den Dienst zu starten und eine neue Verbindung herzustellen. Diese Einstellung kann im Webstatus geändert werden.</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="133"/>
|
||||||
|
<source>Information</source>
|
||||||
|
<translation>Information</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpicommander.py" line="133"/>
|
||||||
|
<source>Can not connect to RevPiPyLoad service through SSH tunnel!
|
||||||
|
|
||||||
|
We are trying to activate this service now and reconnect. The settings can be changed at any time via 'webstatus'.</source>
|
||||||
|
<translation>Vielleicht läuft der RevPiPyLoad Dienst nicht.
|
||||||
|
|
||||||
|
Wir versuchen diesen Dienst jetzt zu aktivieren und verbinden uns neu. Die Einstellungen können über 'Webstatus' jederzeit geändert werden.</translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>RevPiFiles</name>
|
<name>RevPiFiles</name>
|
||||||
@@ -547,6 +791,11 @@ Wählen Sie 'Ja' zum Überschreiben, 'Nein' um nur fehlende
|
|||||||
<source>Choose a local directory first.</source>
|
<source>Choose a local directory first.</source>
|
||||||
<translation>Lokales Verzeichnis wählen.</translation>
|
<translation>Lokales Verzeichnis wählen.</translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpifiles.py" line="127"/>
|
||||||
|
<source>File transfer...</source>
|
||||||
|
<translation>Dateiübertragung...</translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>RevPiInfo</name>
|
<name>RevPiInfo</name>
|
||||||
@@ -572,12 +821,12 @@ Wählen Sie 'Ja' zum Überschreiben, 'Nein' um nur fehlende
|
|||||||
<context>
|
<context>
|
||||||
<name>RevPiOption</name>
|
<name>RevPiOption</name>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="300"/>
|
<location filename="../revpioption.py" line="342"/>
|
||||||
<source>Question</source>
|
<source>Question</source>
|
||||||
<translation>Frage</translation>
|
<translation>Frage</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="149"/>
|
<location filename="../revpioption.py" line="185"/>
|
||||||
<source>The settings will be set on the Revolution Pi now.
|
<source>The settings will be set on the Revolution Pi now.
|
||||||
|
|
||||||
ACL changes and service settings are applied immediately.</source>
|
ACL changes and service settings are applied immediately.</source>
|
||||||
@@ -586,76 +835,76 @@ ACL changes and service settings are applied immediately.</source>
|
|||||||
Berechtigungseinstellungen werden sofort gesetzt.</translation>
|
Berechtigungseinstellungen werden sofort gesetzt.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="196"/>
|
<location filename="../revpioption.py" line="232"/>
|
||||||
<source>Error</source>
|
<source>Error</source>
|
||||||
<translation>Fehler</translation>
|
<translation>Fehler</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="196"/>
|
<location filename="../revpioption.py" line="232"/>
|
||||||
<source>The settings could not be saved on the Revolution Pi!
|
<source>The settings could not be saved on the Revolution Pi!
|
||||||
Try to save the values one mor time and check the log files of RevPiPyLoad if the error rises again.</source>
|
Try to save the values one mor time and check the log files of RevPiPyLoad if the error rises again.</source>
|
||||||
<translation>Die Einstellungen konnten nicht auf dem Revolution Pi gespeichert werden!
|
<translation>Die Einstellungen konnten nicht auf dem Revolution Pi gespeichert werden!
|
||||||
Versuche es erneut und prüfe die Logdateien von RevPiPyLoad, wenn der Fehler erneut auftritt.</translation>
|
Versuche es erneut und prüfe die Logdateien von RevPiPyLoad, wenn der Fehler erneut auftritt.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="206"/>
|
<location filename="../revpioption.py" line="242"/>
|
||||||
<source>Do you really want to quit?
|
<source>Do you really want to quit?
|
||||||
Unsaved changes will be lost.</source>
|
Unsaved changes will be lost.</source>
|
||||||
<translation>Soll das Fenster wirklich geschlossen werden?
|
<translation>Soll das Fenster wirklich geschlossen werden?
|
||||||
Ungesicherte Änderungen gehen verloren.</translation>
|
Ungesicherte Änderungen gehen verloren.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="249"/>
|
<location filename="../revpioption.py" line="291"/>
|
||||||
<source>running</source>
|
<source>running</source>
|
||||||
<translation>läuft</translation>
|
<translation>läuft</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="249"/>
|
<location filename="../revpioption.py" line="291"/>
|
||||||
<source>stopped</source>
|
<source>stopped</source>
|
||||||
<translation>angehalten</translation>
|
<translation>angehalten</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="240"/>
|
<location filename="../revpioption.py" line="282"/>
|
||||||
<source>The MQTT service is not available on your RevPiPyLoad version.</source>
|
<source>The MQTT service is not available on your RevPiPyLoad version.</source>
|
||||||
<translation>MQTT ist in der RevPiPyLoad Version nicht verfügbar.</translation>
|
<translation>MQTT ist in der RevPiPyLoad Version nicht verfügbar.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="280"/>
|
<location filename="../revpioption.py" line="322"/>
|
||||||
<source>read only</source>
|
<source>read only</source>
|
||||||
<translation>Nur lesen</translation>
|
<translation>Nur lesen</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="280"/>
|
<location filename="../revpioption.py" line="322"/>
|
||||||
<source>read and write</source>
|
<source>read and write</source>
|
||||||
<translation>lesen und schreiben</translation>
|
<translation>lesen und schreiben</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="300"/>
|
<location filename="../revpioption.py" line="342"/>
|
||||||
<source>Are you sure you want to deactivate the XML-RPC server? You will NOT be able to access the Revolution Pi with this program after saving the options!</source>
|
<source>Are you sure you want to deactivate the XML-RPC server? You will NOT be able to access the Revolution Pi with this program after saving the options!</source>
|
||||||
<translation>Willst du den XML-RPC Server wirklich deaktivieren? Du wirst dich NICHT mehr mit diesem Programm zum Revolution Pi verbinden können!</translation>
|
<translation>Willst du den XML-RPC Server wirklich deaktivieren? Du wirst dich NICHT mehr mit diesem Programm zum Revolution Pi verbinden können!</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="312"/>
|
<location filename="../revpioption.py" line="354"/>
|
||||||
<source>Start/Stop PLC program and read logs</source>
|
<source>Start/Stop PLC program and read logs</source>
|
||||||
<translation>SPS Programm starten/stoppen und Logs lesen</translation>
|
<translation>SPS Programm starten/stoppen und Logs lesen</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="312"/>
|
<location filename="../revpioption.py" line="354"/>
|
||||||
<source>+ read IOs in watch mode</source>
|
<source>+ read IOs in watch mode</source>
|
||||||
<translation>+ EAs in SPS Betrachter lesen</translation>
|
<translation>+ EAs in SPS Betrachter lesen</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="312"/>
|
<location filename="../revpioption.py" line="354"/>
|
||||||
<source>+ read properties and download PLC program</source>
|
<source>+ read properties and download PLC program</source>
|
||||||
<translation>+ Einstellungen lesen und SPS Programm herunterladen</translation>
|
<translation>+ Einstellungen lesen und SPS Programm herunterladen</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="312"/>
|
<location filename="../revpioption.py" line="354"/>
|
||||||
<source>+ upload PLC program</source>
|
<source>+ upload PLC program</source>
|
||||||
<translation>+ SPS Programm hochladen</translation>
|
<translation>+ SPS Programm hochladen</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpioption.py" line="312"/>
|
<location filename="../revpioption.py" line="354"/>
|
||||||
<source>+ set properties</source>
|
<source>+ set properties</source>
|
||||||
<translation>+ Einstellungen ändern</translation>
|
<translation>+ Einstellungen ändern</translation>
|
||||||
</message>
|
</message>
|
||||||
@@ -668,17 +917,31 @@ Ungesicherte Änderungen gehen verloren.</translation>
|
|||||||
<translation type="obsolete">Neue Verbindung</translation>
|
<translation type="obsolete">Neue Verbindung</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpiplclist.py" line="102"/>
|
<location filename="../revpiplclist.py" line="301"/>
|
||||||
<source>Question</source>
|
<source>Question</source>
|
||||||
<translation>Frage</translation>
|
<translation>Frage</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../revpiplclist.py" line="102"/>
|
<location filename="../revpiplclist.py" line="132"/>
|
||||||
<source>Do you really want to quit?
|
<source>Do you really want to quit?
|
||||||
Unsaved changes will be lost.</source>
|
Unsaved changes will be lost.</source>
|
||||||
<translation>Soll das Fenster wirklich geschlossen werden?
|
<translation>Soll das Fenster wirklich geschlossen werden?
|
||||||
Ungesicherte Änderungen gehen verloren.</translation>
|
Ungesicherte Änderungen gehen verloren.</translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpiplclist.py" line="301"/>
|
||||||
|
<source>If you remote this folder, all containing elements will be removed, too.
|
||||||
|
|
||||||
|
Do you want to delete folder and all elements?</source>
|
||||||
|
<translation>Wird dieser Ordner gelöscht, betrifft dies auch alle Elemente im Ordner.
|
||||||
|
|
||||||
|
Wollen sie den Ordner und alle Elemente löschen?</translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../revpiplclist.py" line="341"/>
|
||||||
|
<source>New folder</source>
|
||||||
|
<translation>Neuer Ordner</translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>RevPiProgram</name>
|
<name>RevPiProgram</name>
|
||||||
@@ -955,12 +1218,12 @@ Dies ist kein Fehler, wenn das SPS Startprogramm bereits auf dem Rev Pi ist. Pr
|
|||||||
<context>
|
<context>
|
||||||
<name>SSHAuth</name>
|
<name>SSHAuth</name>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../sshauth.py" line="49"/>
|
<location filename="../sshauth.py" line="51"/>
|
||||||
<source>Could not save password</source>
|
<source>Could not save password</source>
|
||||||
<translation>Konnte Kennwort nicht speichern</translation>
|
<translation>Konnte Kennwort nicht speichern</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../sshauth.py" line="49"/>
|
<location filename="../sshauth.py" line="51"/>
|
||||||
<source>Could not save password to operating systems password save.
|
<source>Could not save password to operating systems password save.
|
||||||
|
|
||||||
Maybe your operating system does not support saving passwords. This could be due to missing libraries or programs.
|
Maybe your operating system does not support saving passwords. This could be due to missing libraries or programs.
|
||||||
@@ -976,12 +1239,12 @@ Dies ist kein Fehler von RevPi Commander.</translation>
|
|||||||
<context>
|
<context>
|
||||||
<name>Simulator</name>
|
<name>Simulator</name>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../simulator.py" line="79"/>
|
<location filename="../simulator.py" line="80"/>
|
||||||
<source>Select downloaded piCtory file...</source>
|
<source>Select downloaded piCtory file...</source>
|
||||||
<translation>Heruntergeladene piCtory Datei auswählen...</translation>
|
<translation>Heruntergeladene piCtory Datei auswählen...</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../simulator.py" line="79"/>
|
<location filename="../simulator.py" line="80"/>
|
||||||
<source>piCtory file (*.rsc);;All files (*.*)</source>
|
<source>piCtory file (*.rsc);;All files (*.*)</source>
|
||||||
<translation>piCtory Datei (*.rsc);;Alle Dateien (*.*)</translation>
|
<translation>piCtory Datei (*.rsc);;Alle Dateien (*.*)</translation>
|
||||||
</message>
|
</message>
|
||||||
@@ -1049,7 +1312,7 @@ Dies ist kein Fehler von RevPi Commander.</translation>
|
|||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/backgroundworker.ui" line="14"/>
|
<location filename="../../../ui_dev/backgroundworker.ui" line="14"/>
|
||||||
<source>File transfer...</source>
|
<source>File transfer...</source>
|
||||||
<translation>Dateiübertragung...</translation>
|
<translation type="obsolete">Dateiübertragung...</translation>
|
||||||
</message>
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
@@ -1060,12 +1323,12 @@ Dies ist kein Fehler von RevPi Commander.</translation>
|
|||||||
<translation>Revolution Pi Verbindungen</translation>
|
<translation>Revolution Pi Verbindungen</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/revpiplclist.ui" line="194"/>
|
<location filename="../../../ui_dev/revpiplclist.ui" line="197"/>
|
||||||
<source>Connection name</source>
|
<source>Connection name</source>
|
||||||
<translation>Verbindungsname</translation>
|
<translation>Verbindungsname</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/revpiplclist.ui" line="199"/>
|
<location filename="../../../ui_dev/revpiplclist.ui" line="202"/>
|
||||||
<source>Address</source>
|
<source>Address</source>
|
||||||
<translation>Adresse</translation>
|
<translation>Adresse</translation>
|
||||||
</message>
|
</message>
|
||||||
@@ -1110,22 +1373,22 @@ Dies ist kein Fehler von RevPi Commander.</translation>
|
|||||||
<translation>Verbindung</translation>
|
<translation>Verbindung</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/revpiplclist.ui" line="122"/>
|
<location filename="../../../ui_dev/revpiplclist.ui" line="125"/>
|
||||||
<source>Over SSH</source>
|
<source>Over SSH</source>
|
||||||
<translation>Über SSH</translation>
|
<translation>Über SSH</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/revpiplclist.ui" line="128"/>
|
<location filename="../../../ui_dev/revpiplclist.ui" line="131"/>
|
||||||
<source>Connect over SSH tunnel:</source>
|
<source>Connect over SSH tunnel:</source>
|
||||||
<translation>Über SSH Tunnel verbinden:</translation>
|
<translation>Über SSH Tunnel verbinden:</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/revpiplclist.ui" line="145"/>
|
<location filename="../../../ui_dev/revpiplclist.ui" line="148"/>
|
||||||
<source>SSH port:</source>
|
<source>SSH port:</source>
|
||||||
<translation>SSH Port:</translation>
|
<translation>SSH Port:</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/revpiplclist.ui" line="162"/>
|
<location filename="../../../ui_dev/revpiplclist.ui" line="165"/>
|
||||||
<source>SSH user name:</source>
|
<source>SSH user name:</source>
|
||||||
<translation>SSH Benutzername:</translation>
|
<translation>SSH Benutzername:</translation>
|
||||||
</message>
|
</message>
|
||||||
@@ -1207,7 +1470,9 @@ Dies ist kein Fehler von RevPi Commander.</translation>
|
|||||||
<source>The base topic is the first part of any mqtt topic, the Revolution Pi will publish. You can use any character includig '/' to structure the messages on your broker.
|
<source>The base topic is the first part of any mqtt topic, the Revolution Pi will publish. You can use any character includig '/' to structure the messages on your broker.
|
||||||
|
|
||||||
For example: revpi0000/data</source>
|
For example: revpi0000/data</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation>Der Basistopic wird allen MQTT Topics vorangestellt, welche der Revolution Pi veröffentlicht. Es können alle Zeichen inklusive '/' verwendet werden, um die Nachrichten auf dem Broker zu strukturieren.
|
||||||
|
|
||||||
|
Zum Beispiel: revpi0000/data</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/mqttmanager.ui" line="80"/>
|
<location filename="../../../ui_dev/mqttmanager.ui" line="80"/>
|
||||||
@@ -1224,7 +1489,9 @@ For example: revpi0000/data</source>
|
|||||||
<source>The Revolution Pi will subscribe a topic on which your mqtt client can publish messages with the new io value as payload.
|
<source>The Revolution Pi will subscribe a topic on which your mqtt client can publish messages with the new io value as payload.
|
||||||
|
|
||||||
Publish values with topic: [basetopic]/set/[outputname]</source>
|
Publish values with topic: [basetopic]/set/[outputname]</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation>Der Revolution Pi abonniert ein Topic, auf dem die MQTT Clients über den Inhalt einer Nachricht einen neuen Ausgangswert setzen können.
|
||||||
|
|
||||||
|
Sende Werte mit Topic: [basistopic]/set/[ausgangsname]</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/mqttmanager.ui" line="175"/>
|
<location filename="../../../ui_dev/mqttmanager.ui" line="175"/>
|
||||||
@@ -1491,22 +1758,17 @@ Publish values with topic: [basetopic]/set/[outputname]</source>
|
|||||||
<translation>RevPi Python SPS - Commander</translation>
|
<translation>RevPi Python SPS - Commander</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/revpiinfo.ui" line="69"/>
|
<location filename="../../../ui_dev/revpiinfo.ui" line="86"/>
|
||||||
<source><html><head/><body><p><a href="https://revpimodio.org/"><span style=" text-decoration: underline; color:#0000ff;">https://revpimodio.org/</span></a></p></body></html></source>
|
|
||||||
<translation type="unfinished"></translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../../../ui_dev/revpiinfo.ui" line="96"/>
|
|
||||||
<source>Version:</source>
|
<source>Version:</source>
|
||||||
<translation></translation>
|
<translation></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/revpiinfo.ui" line="126"/>
|
<location filename="../../../ui_dev/revpiinfo.ui" line="116"/>
|
||||||
<source>RevPiModIO, RevPiPyLoad and RevPiPyControl are community driven projects. They are all free and open source software.
|
<source>RevPiModIO, RevPiPyLoad and RevPiPyControl are community driven projects. They are all free and open source software.
|
||||||
All of them comes with ABSOLUTELY NO WARRANTY, to the extent permitted by
|
All of them comes with ABSOLUTELY NO WARRANTY, to the extent permitted by
|
||||||
applicable law.
|
applicable law.
|
||||||
|
|
||||||
(c) Sven Sager, License: GPLv3</source>
|
(c) Sven Sager, License: GPLv2</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
</context>
|
</context>
|
||||||
@@ -1565,7 +1827,7 @@ applicable law.
|
|||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/avahisearch.ui" line="140"/>
|
<location filename="../../../ui_dev/avahisearch.ui" line="140"/>
|
||||||
<source>Connect via SSH (recommended)</source>
|
<source>Connect via SSH (recommended)</source>
|
||||||
<translation>Über SSH verbinden (empfholen)</translation>
|
<translation>Über SSH verbinden (empfohlen)</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/avahisearch.ui" line="143"/>
|
<location filename="../../../ui_dev/avahisearch.ui" line="143"/>
|
||||||
@@ -1684,25 +1946,30 @@ applicable law.
|
|||||||
<translation>SSH Authentifizierung</translation>
|
<translation>SSH Authentifizierung</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/sshauth.ui" line="26"/>
|
<location filename="../../../ui_dev/sshauth.ui" line="29"/>
|
||||||
<source>SSH username:</source>
|
<source>SSH username:</source>
|
||||||
<translation>SSH Benutzername:</translation>
|
<translation>SSH Benutzername:</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/sshauth.ui" line="33"/>
|
<location filename="../../../ui_dev/sshauth.ui" line="36"/>
|
||||||
<source>SSH password:</source>
|
<source>SSH password:</source>
|
||||||
<translation>SSH Passwort:</translation>
|
<translation>SSH Passwort:</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/sshauth.ui" line="53"/>
|
<location filename="../../../ui_dev/sshauth.ui" line="56"/>
|
||||||
<source>Username and password will be saved in secured operating systems's password storage.</source>
|
<source>Username and password will be saved in secured operating systems's password storage.</source>
|
||||||
<translation>Benutzername und Kennwort werden im Passwortspeicher vom Betriebssystem gesichert.</translation>
|
<translation>Benutzername und Kennwort werden im Passwortspeicher vom Betriebssystem gesichert.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../../../ui_dev/sshauth.ui" line="56"/>
|
<location filename="../../../ui_dev/sshauth.ui" line="59"/>
|
||||||
<source>Save username and password</source>
|
<source>Save username and password</source>
|
||||||
<translation>Benutzername und Kennwort merken</translation>
|
<translation>Benutzername und Kennwort merken</translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<location filename="../../../ui_dev/sshauth.ui" line="82"/>
|
||||||
|
<source>Note: The default user for SSH is "pi" which differs from the web configuration. You can find the password on the sticker on the device.</source>
|
||||||
|
<translation>Hinweis: Der Standardbenutzer für SSH ist "pi" dies weicht von der Web-Konfiguration ab. Das Kennwort finden sie auf dem Aufkleber am Gerät.</translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>wid_debugcontrol</name>
|
<name>wid_debugcontrol</name>
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
from PyQt5 import QtGui, QtWidgets
|
from PyQt5 import QtGui, QtWidgets
|
||||||
|
|
||||||
@@ -15,7 +15,7 @@ class MqttManager(QtWidgets.QDialog, Ui_diag_mqtt):
|
|||||||
"""MQTT settings for option window."""
|
"""MQTT settings for option window."""
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
super(MqttManager, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
self.setFixedSize(self.size())
|
self.setFixedSize(self.size())
|
||||||
|
|
||||||
@@ -61,7 +61,7 @@ class MqttManager(QtWidgets.QDialog, Ui_diag_mqtt):
|
|||||||
return True
|
return True
|
||||||
|
|
||||||
def accept(self) -> None:
|
def accept(self) -> None:
|
||||||
"""Save values to master dict."""
|
"""Save values to value dict."""
|
||||||
self.dc["mqttbasetopic"] = self.txt_basetopic.text()
|
self.dc["mqttbasetopic"] = self.txt_basetopic.text()
|
||||||
self.dc["mqttsendinterval"] = self.sbx_sendinterval.value()
|
self.dc["mqttsendinterval"] = self.sbx_sendinterval.value()
|
||||||
self.dc["mqttsend_on_event"] = int(self.cbx_send_on_event.isChecked())
|
self.dc["mqttsend_on_event"] = int(self.cbx_send_on_event.isChecked())
|
||||||
@@ -72,7 +72,7 @@ class MqttManager(QtWidgets.QDialog, Ui_diag_mqtt):
|
|||||||
self.dc["mqttusername"] = self.txt_username.text()
|
self.dc["mqttusername"] = self.txt_username.text()
|
||||||
self.dc["mqttpassword"] = self.txt_password.text()
|
self.dc["mqttpassword"] = self.txt_password.text()
|
||||||
self.dc["mqttclient_id"] = self.txt_client_id.text()
|
self.dc["mqttclient_id"] = self.txt_client_id.text()
|
||||||
super(MqttManager, self).accept()
|
super().accept()
|
||||||
|
|
||||||
def closeEvent(self, a0: QtGui.QCloseEvent) -> None:
|
def closeEvent(self, a0: QtGui.QCloseEvent) -> None:
|
||||||
if self._changesdone():
|
if self._changesdone():
|
||||||
@@ -95,12 +95,12 @@ class MqttManager(QtWidgets.QDialog, Ui_diag_mqtt):
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
return QtWidgets.QDialog.Rejected
|
return QtWidgets.QDialog.Rejected
|
||||||
return super(MqttManager, self).exec()
|
return super().exec()
|
||||||
|
|
||||||
def reject(self) -> None:
|
def reject(self) -> None:
|
||||||
"""Reject settings."""
|
"""Reject settings."""
|
||||||
self._load_settings()
|
self._load_settings()
|
||||||
super(MqttManager, self).reject()
|
super().reject()
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def read_only(self):
|
def read_only(self):
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""Global program initialization."""
|
"""Global program initialization."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "LGPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
import sys
|
import sys
|
||||||
@@ -89,11 +89,12 @@ parser.add_argument(
|
|||||||
"-f", "--logfile", dest="logfile",
|
"-f", "--logfile", dest="logfile",
|
||||||
help="Save log entries to this file"
|
help="Save log entries to this file"
|
||||||
)
|
)
|
||||||
|
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"-v", "--verbose", action="count", dest="verbose", default=0,
|
"-v", "--verbose", action="count", dest="verbose", default=0,
|
||||||
help="Switch on verbose logging"
|
help="Switch on verbose logging"
|
||||||
)
|
)
|
||||||
|
# The __main__ script will process the version number argument
|
||||||
|
parser.add_argument("--version", action="store_true", help="Print version number of program and exit")
|
||||||
pargs = parser.parse_args()
|
pargs = parser.parse_args()
|
||||||
|
|
||||||
# Check important objects and set to default if they do not exists
|
# Check important objects and set to default if they do not exists
|
||||||
|
|||||||
@@ -4,35 +4,55 @@
|
|||||||
|
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2018 Sven Sager"
|
__copyright__ = "Copyright (C) 2018 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
__version__ = "0.9.10rc2"
|
|
||||||
|
|
||||||
import webbrowser
|
import webbrowser
|
||||||
from os.path import dirname, join
|
from os.path import dirname, join
|
||||||
|
|
||||||
from PyQt5 import QtCore, QtGui, QtWidgets
|
from PyQt5 import QtCore, QtGui, QtWidgets
|
||||||
|
|
||||||
|
from . import __version__
|
||||||
from . import helper
|
from . import helper
|
||||||
from . import proginit as pi
|
from . import proginit as pi
|
||||||
from . import revpilogfile
|
from . import revpilogfile
|
||||||
from .avahisearch import AvahiSearch
|
from .avahisearch import AvahiSearch
|
||||||
|
from .backgroundworker import BackgroundWaiter
|
||||||
from .debugcontrol import DebugControl
|
from .debugcontrol import DebugControl
|
||||||
from .helper import RevPiSettings
|
from .helper import ConnectionFail, RevPiSettings
|
||||||
from .revpifiles import RevPiFiles
|
from .revpifiles import RevPiFiles
|
||||||
from .revpiinfo import RevPiInfo
|
from .revpiinfo import RevPiInfo
|
||||||
from .revpioption import RevPiOption
|
from .revpioption import RevPiOption
|
||||||
from .revpiplclist import RevPiPlcList
|
from .revpiplclist import RevPiPlcList
|
||||||
from .revpiprogram import RevPiProgram
|
from .revpiprogram import RevPiProgram
|
||||||
from .simulator import Simulator
|
from .simulator import Simulator
|
||||||
|
from .sshauth import SSHAuth
|
||||||
from .ui.revpicommander_ui import Ui_win_revpicommander
|
from .ui.revpicommander_ui import Ui_win_revpicommander
|
||||||
|
|
||||||
|
|
||||||
|
class ConnectingPyload(QtCore.QThread):
|
||||||
|
"""
|
||||||
|
Try to establish a connection in background.
|
||||||
|
|
||||||
|
The pyload_connect function will emit signals for error or successful connect. This
|
||||||
|
signals will be used to show error messages and return to this function, if the
|
||||||
|
authentication failed.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, revpi_settings: RevPiSettings, ssh_password="", parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
self._revpi_settings = revpi_settings
|
||||||
|
self._ssh_password = ssh_password
|
||||||
|
|
||||||
|
def run(self) -> None:
|
||||||
|
helper.cm.pyload_connect(self._revpi_settings, self._ssh_password)
|
||||||
|
|
||||||
|
|
||||||
class RevPiCommander(QtWidgets.QMainWindow, Ui_win_revpicommander):
|
class RevPiCommander(QtWidgets.QMainWindow, Ui_win_revpicommander):
|
||||||
"""Main application of RevPiCommander."""
|
"""Main application of RevPiCommander."""
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
"""Init main program."""
|
"""Init main program."""
|
||||||
super(RevPiCommander, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
|
|
||||||
self.wid_debugcontrol = None # type: DebugControl
|
self.wid_debugcontrol = None # type: DebugControl
|
||||||
@@ -56,6 +76,7 @@ class RevPiCommander(QtWidgets.QMainWindow, Ui_win_revpicommander):
|
|||||||
|
|
||||||
self.btn_plc_logs.clicked.connect(self.on_act_logs_triggered)
|
self.btn_plc_logs.clicked.connect(self.on_act_logs_triggered)
|
||||||
|
|
||||||
|
helper.cm.connect_error.connect(self.on_cm_connect_error)
|
||||||
helper.cm.connection_disconnected.connect(self.on_cm_connection_disconnected)
|
helper.cm.connection_disconnected.connect(self.on_cm_connection_disconnected)
|
||||||
helper.cm.connection_disconnecting.connect(self.on_cm_connection_disconnecting)
|
helper.cm.connection_disconnecting.connect(self.on_cm_connection_disconnecting)
|
||||||
helper.cm.connection_established.connect(self.on_cm_connection_established)
|
helper.cm.connection_established.connect(self.on_cm_connection_established)
|
||||||
@@ -97,6 +118,33 @@ class RevPiCommander(QtWidgets.QMainWindow, Ui_win_revpicommander):
|
|||||||
# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
|
# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
|
||||||
# region # REGION: Connection management
|
# region # REGION: Connection management
|
||||||
|
|
||||||
|
@QtCore.pyqtSlot(str, str, ConnectionFail, RevPiSettings)
|
||||||
|
def on_cm_connect_error(self, title: str, text: str, fail_code: ConnectionFail, revpi_settings: RevPiSettings):
|
||||||
|
"""
|
||||||
|
Slot to get information of pyload_connect connection errors.
|
||||||
|
|
||||||
|
:param title: Title of error message
|
||||||
|
:param text: Text of error message
|
||||||
|
:param fail_code: Type of error
|
||||||
|
:param revpi_settings: Settings of the revpi with the error
|
||||||
|
"""
|
||||||
|
if fail_code is ConnectionFail.NO_XML_RPC_VIA_TUNNEL:
|
||||||
|
# If RevPiPyLoad is not running, we can try to activate it via ssh
|
||||||
|
QtWidgets.QMessageBox.information(
|
||||||
|
self, self.tr("Information"), self.tr(
|
||||||
|
"Can not connect to RevPiPyLoad service through SSH tunnel!\n\n"
|
||||||
|
"We are trying to activate this service now and reconnect. The settings can be "
|
||||||
|
"changed at any time via 'webstatus'."
|
||||||
|
),
|
||||||
|
)
|
||||||
|
revpi_settings.ssh_enable_revpipyload = True
|
||||||
|
self._pyload_connect(revpi_settings, False)
|
||||||
|
else:
|
||||||
|
QtWidgets.QMessageBox.critical(self, title, text)
|
||||||
|
if fail_code is ConnectionFail.SSH_AUTH:
|
||||||
|
# On failed credentials, we try to connect again and remove password form keychain, if exists
|
||||||
|
self._pyload_connect(revpi_settings, True)
|
||||||
|
|
||||||
@QtCore.pyqtSlot(str)
|
@QtCore.pyqtSlot(str)
|
||||||
def on_cm_connection_error_observed(self, message: str):
|
def on_cm_connection_error_observed(self, message: str):
|
||||||
"""
|
"""
|
||||||
@@ -179,6 +227,8 @@ class RevPiCommander(QtWidgets.QMainWindow, Ui_win_revpicommander):
|
|||||||
parent_menu = self.men_connections
|
parent_menu = self.men_connections
|
||||||
|
|
||||||
display_name = settings.name
|
display_name = settings.name
|
||||||
|
if settings.name != settings.address:
|
||||||
|
display_name += " [{0}]".format(settings.address)
|
||||||
if settings.ssh_use_tunnel:
|
if settings.ssh_use_tunnel:
|
||||||
display_name += " (SSH)"
|
display_name += " (SSH)"
|
||||||
|
|
||||||
@@ -188,6 +238,54 @@ class RevPiCommander(QtWidgets.QMainWindow, Ui_win_revpicommander):
|
|||||||
act.setToolTip("{0}:{1}".format(settings.address, settings.port))
|
act.setToolTip("{0}:{1}".format(settings.address, settings.port))
|
||||||
parent_menu.addAction(act)
|
parent_menu.addAction(act)
|
||||||
|
|
||||||
|
def _pyload_connect(self, revpi_settings: RevPiSettings, remove_saved_ssh_password=False):
|
||||||
|
"""
|
||||||
|
Try to async establish a connection to Revolution Pi.
|
||||||
|
|
||||||
|
:param revpi_settings: RevPi settings object
|
||||||
|
:param remove_saved_ssh_password: Remove password from keychain
|
||||||
|
"""
|
||||||
|
ssh_password = ""
|
||||||
|
|
||||||
|
diag_connecting = BackgroundWaiter(
|
||||||
|
revpi_settings.timeout,
|
||||||
|
self.tr("Establish a connection to the Revolution Pi..."),
|
||||||
|
self,
|
||||||
|
self.tr("Revolution Pi connected!"),
|
||||||
|
)
|
||||||
|
helper.cm.connection_established.connect(diag_connecting.requestInterruption)
|
||||||
|
|
||||||
|
if revpi_settings.ssh_use_tunnel:
|
||||||
|
diag_ssh_auth = SSHAuth(
|
||||||
|
revpi_settings.ssh_user,
|
||||||
|
"{0}.{1}_{2}".format(
|
||||||
|
helper.settings.applicationName(),
|
||||||
|
helper.settings.organizationName(),
|
||||||
|
revpi_settings.internal_id),
|
||||||
|
self,
|
||||||
|
)
|
||||||
|
|
||||||
|
if remove_saved_ssh_password and revpi_settings.ssh_saved_password:
|
||||||
|
diag_ssh_auth.remove_saved_password()
|
||||||
|
revpi_settings.ssh_saved_password = False
|
||||||
|
|
||||||
|
# Doesn't matter what user selects, we have to save settings to sync keychain and values
|
||||||
|
if diag_ssh_auth.exec() != QtWidgets.QDialog.Accepted:
|
||||||
|
revpi_settings.save_settings()
|
||||||
|
return
|
||||||
|
|
||||||
|
revpi_settings.ssh_saved_password = diag_ssh_auth.in_keyring
|
||||||
|
revpi_settings.ssh_user = diag_ssh_auth.username
|
||||||
|
ssh_password = diag_ssh_auth.password
|
||||||
|
revpi_settings.save_settings()
|
||||||
|
|
||||||
|
# Connect in background and show the connecting dialog to user
|
||||||
|
th_connecting = ConnectingPyload(revpi_settings, ssh_password, self)
|
||||||
|
th_connecting.finished.connect(diag_connecting.requestInterruption)
|
||||||
|
th_connecting.start()
|
||||||
|
|
||||||
|
diag_connecting.exec_dialog(self.tr("Connecting..."), False)
|
||||||
|
|
||||||
@QtCore.pyqtSlot()
|
@QtCore.pyqtSlot()
|
||||||
def on_act_connections_triggered(self):
|
def on_act_connections_triggered(self):
|
||||||
"""Edit saved connections to Revolution Pi devices."""
|
"""Edit saved connections to Revolution Pi devices."""
|
||||||
@@ -202,7 +300,7 @@ class RevPiCommander(QtWidgets.QMainWindow, Ui_win_revpicommander):
|
|||||||
if self.diag_search.just_save:
|
if self.diag_search.just_save:
|
||||||
self.diag_connections.exec_with_presets(self.diag_search.connect_settings)
|
self.diag_connections.exec_with_presets(self.diag_search.connect_settings)
|
||||||
else:
|
else:
|
||||||
helper.cm.pyload_connect(self.diag_search.connect_settings, self)
|
self._pyload_connect(self.diag_search.connect_settings)
|
||||||
|
|
||||||
self._load_men_connections()
|
self._load_men_connections()
|
||||||
|
|
||||||
@@ -218,7 +316,7 @@ class RevPiCommander(QtWidgets.QMainWindow, Ui_win_revpicommander):
|
|||||||
configrsc_file = helper.settings.value("simulator/configrsc", "", str)
|
configrsc_file = helper.settings.value("simulator/configrsc", "", str)
|
||||||
procimg_file = helper.settings.value("simulator/procimg", "", str)
|
procimg_file = helper.settings.value("simulator/procimg", "", str)
|
||||||
|
|
||||||
if helper.cm.pyload_simulate(configrsc_file, procimg_file, diag.cbx_stop_remove.isChecked()):
|
if helper.cm.pyload_simulate(configrsc_file, procimg_file, diag.clean_procimg):
|
||||||
QtWidgets.QMessageBox.information(
|
QtWidgets.QMessageBox.information(
|
||||||
self, self.tr("Simulator started..."), self.tr(
|
self, self.tr("Simulator started..."), self.tr(
|
||||||
"The simulator is running!\n\nYou can work with this simulator if your call "
|
"The simulator is running!\n\nYou can work with this simulator if your call "
|
||||||
@@ -318,8 +416,8 @@ class RevPiCommander(QtWidgets.QMainWindow, Ui_win_revpicommander):
|
|||||||
@QtCore.pyqtSlot()
|
@QtCore.pyqtSlot()
|
||||||
def on_act_pictory_triggered(self):
|
def on_act_pictory_triggered(self):
|
||||||
"""Open piCtory in default browser of operating system."""
|
"""Open piCtory in default browser of operating system."""
|
||||||
if helper.cm.address:
|
if helper.cm.settings.address:
|
||||||
webbrowser.open("http://{0}/".format(helper.cm.address))
|
webbrowser.open("http://{0}/".format(helper.cm.settings.address))
|
||||||
|
|
||||||
@QtCore.pyqtSlot()
|
@QtCore.pyqtSlot()
|
||||||
def on_act_reset_triggered(self):
|
def on_act_reset_triggered(self):
|
||||||
@@ -361,7 +459,7 @@ class RevPiCommander(QtWidgets.QMainWindow, Ui_win_revpicommander):
|
|||||||
@QtCore.pyqtSlot(QtWidgets.QAction)
|
@QtCore.pyqtSlot(QtWidgets.QAction)
|
||||||
def on_men_connections_triggered(self, action: QtWidgets.QAction):
|
def on_men_connections_triggered(self, action: QtWidgets.QAction):
|
||||||
"""A connection is selected in the men_connections menu."""
|
"""A connection is selected in the men_connections menu."""
|
||||||
helper.cm.pyload_connect(action.data(), self)
|
self._pyload_connect(action.data())
|
||||||
|
|
||||||
@QtCore.pyqtSlot()
|
@QtCore.pyqtSlot()
|
||||||
def on_act_webpage_triggered(self):
|
def on_act_webpage_triggered(self):
|
||||||
@@ -464,6 +562,7 @@ class RevPiCommander(QtWidgets.QMainWindow, Ui_win_revpicommander):
|
|||||||
|
|
||||||
|
|
||||||
def main() -> int:
|
def main() -> int:
|
||||||
|
"""Entry point for RevPiCommander."""
|
||||||
from sys import argv
|
from sys import argv
|
||||||
|
|
||||||
if hasattr(QtCore.Qt, 'AA_EnableHighDpiScaling'):
|
if hasattr(QtCore.Qt, 'AA_EnableHighDpiScaling'):
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""File manager for up und download PLC program."""
|
"""File manager for up und download PLC program."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
import gzip
|
import gzip
|
||||||
import os
|
import os
|
||||||
@@ -26,7 +26,7 @@ class NodeType(IntEnum):
|
|||||||
class UploadFiles(BackgroundWorker):
|
class UploadFiles(BackgroundWorker):
|
||||||
|
|
||||||
def __init__(self, file_list: list, parent):
|
def __init__(self, file_list: list, parent):
|
||||||
super(UploadFiles, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.ec = 1
|
self.ec = 1
|
||||||
self.file_list = file_list
|
self.file_list = file_list
|
||||||
self.plc_program_included = False # Will be True, when opt_program was found in files
|
self.plc_program_included = False # Will be True, when opt_program was found in files
|
||||||
@@ -76,7 +76,7 @@ class UploadFiles(BackgroundWorker):
|
|||||||
class RevPiFiles(QtWidgets.QMainWindow, Ui_win_files):
|
class RevPiFiles(QtWidgets.QMainWindow, Ui_win_files):
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
super(RevPiFiles, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
|
|
||||||
self.dc_settings = {}
|
self.dc_settings = {}
|
||||||
@@ -124,7 +124,7 @@ class RevPiFiles(QtWidgets.QMainWindow, Ui_win_files):
|
|||||||
return
|
return
|
||||||
|
|
||||||
uploader = UploadFiles(self.file_list_local(), self)
|
uploader = UploadFiles(self.file_list_local(), self)
|
||||||
if uploader.exec_dialog() == QtWidgets.QDialog.Rejected:
|
if uploader.exec_dialog(self.tr("File transfer...")) == QtWidgets.QDialog.Rejected:
|
||||||
return
|
return
|
||||||
|
|
||||||
if uploader.ec == 0:
|
if uploader.ec == 0:
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""Program information of local an remote system."""
|
"""Program information of local an remote system."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
from PyQt5 import QtCore, QtGui, QtWidgets
|
from PyQt5 import QtCore, QtGui, QtWidgets
|
||||||
|
|
||||||
@@ -14,7 +14,7 @@ class RevPiInfo(QtWidgets.QDialog, Ui_diag_revpiinfo):
|
|||||||
"""Version information window."""
|
"""Version information window."""
|
||||||
|
|
||||||
def __init__(self, version: str, parent=None):
|
def __init__(self, version: str, parent=None):
|
||||||
super(RevPiInfo, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
|
|
||||||
self._debug_load = False
|
self._debug_load = False
|
||||||
@@ -24,11 +24,11 @@ class RevPiInfo(QtWidgets.QDialog, Ui_diag_revpiinfo):
|
|||||||
|
|
||||||
def exec(self) -> int:
|
def exec(self) -> int:
|
||||||
self.lbl_version_pyload.setText(
|
self.lbl_version_pyload.setText(
|
||||||
"{0}.{1}.{2}".format(*helper.cm.pyload_version)
|
helper.cm.pyload_version_str
|
||||||
if helper.cm.connected else "-"
|
if helper.cm.connected else "-"
|
||||||
)
|
)
|
||||||
self._load_lst_files()
|
self._load_lst_files()
|
||||||
return super(RevPiInfo, self).exec()
|
return super().exec()
|
||||||
|
|
||||||
def lbl_version_mousePressEvent(self, a0: QtGui.QMouseEvent):
|
def lbl_version_mousePressEvent(self, a0: QtGui.QMouseEvent):
|
||||||
if a0.button() == QtCore.Qt.MidButton:
|
if a0.button() == QtCore.Qt.MidButton:
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""View log files from Revolution Pi."""
|
"""View log files from Revolution Pi."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
from enum import IntEnum
|
from enum import IntEnum
|
||||||
|
|
||||||
@@ -25,7 +25,7 @@ class DataThread(QtCore.QThread):
|
|||||||
"""log_type, success, text"""
|
"""log_type, success, text"""
|
||||||
|
|
||||||
def __init__(self, parent=None, cycle_time=1000):
|
def __init__(self, parent=None, cycle_time=1000):
|
||||||
super(DataThread, self).__init__(parent)
|
super().__init__(parent)
|
||||||
|
|
||||||
self._cycle_time = cycle_time
|
self._cycle_time = cycle_time
|
||||||
self._paused = True
|
self._paused = True
|
||||||
@@ -113,7 +113,7 @@ class RevPiLogfile(QtWidgets.QMainWindow, Ui_win_revpilogfile):
|
|||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
u"""Init RevPiLogfile-Class."""
|
u"""Init RevPiLogfile-Class."""
|
||||||
super(RevPiLogfile, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
|
|
||||||
self.th_data = DataThread(self)
|
self.th_data = DataThread(self)
|
||||||
@@ -138,11 +138,11 @@ class RevPiLogfile(QtWidgets.QMainWindow, Ui_win_revpilogfile):
|
|||||||
|
|
||||||
def hideEvent(self, a0: QtGui.QHideEvent) -> None:
|
def hideEvent(self, a0: QtGui.QHideEvent) -> None:
|
||||||
self.th_data.pause()
|
self.th_data.pause()
|
||||||
super(RevPiLogfile, self).hideEvent(a0)
|
super().hideEvent(a0)
|
||||||
|
|
||||||
def showEvent(self, a0: QtGui.QShowEvent) -> None:
|
def showEvent(self, a0: QtGui.QShowEvent) -> None:
|
||||||
self.th_data.resume()
|
self.th_data.resume()
|
||||||
super(RevPiLogfile, self).showEvent(a0)
|
super().showEvent(a0)
|
||||||
|
|
||||||
def _load_gui_settings(self):
|
def _load_gui_settings(self):
|
||||||
self.restoreGeometry(helper.settings.value("logfile/geo", b''))
|
self.restoreGeometry(helper.settings.value("logfile/geo", b''))
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""RevPiPyLoad options window."""
|
"""RevPiPyLoad options window."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
from PyQt5 import QtCore, QtGui, QtWidgets
|
from PyQt5 import QtCore, QtGui, QtWidgets
|
||||||
|
|
||||||
@@ -17,14 +17,15 @@ class RevPiOption(QtWidgets.QDialog, Ui_diag_options):
|
|||||||
"""Set options of RevPiPyLoad."""
|
"""Set options of RevPiPyLoad."""
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
super(RevPiOption, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
self.setFixedSize(self.size())
|
self.setFixedSize(self.size())
|
||||||
|
|
||||||
self.dc = {}
|
self.dc = {}
|
||||||
self.acl_plcslave = ""
|
self.acl_plcserver = ""
|
||||||
self.acl_xmlrpc = ""
|
self.acl_xmlrpc = ""
|
||||||
self.mrk_xml_ask = True
|
self.mrk_xml_ask = True
|
||||||
|
self.wrong_names = False
|
||||||
|
|
||||||
self._dict_mqttsettings = {
|
self._dict_mqttsettings = {
|
||||||
"mqttbasetopic": "revpi01",
|
"mqttbasetopic": "revpi01",
|
||||||
@@ -56,7 +57,7 @@ class RevPiOption(QtWidgets.QDialog, Ui_diag_options):
|
|||||||
self.cbx_zeroonerror.setEnabled(allow)
|
self.cbx_zeroonerror.setEnabled(allow)
|
||||||
self.cbb_replace_io.setEnabled(allow)
|
self.cbb_replace_io.setEnabled(allow)
|
||||||
self.txt_replace_io.setEnabled(allow and self.cbb_replace_io.currentIndex() == 3)
|
self.txt_replace_io.setEnabled(allow and self.cbb_replace_io.currentIndex() == 3)
|
||||||
self.cbx_plcslave.setEnabled(allow)
|
self.cbx_plcserver.setEnabled(allow)
|
||||||
self.cbx_mqtt.setEnabled(allow)
|
self.cbx_mqtt.setEnabled(allow)
|
||||||
self.cbx_xmlrpc.setEnabled(allow)
|
self.cbx_xmlrpc.setEnabled(allow)
|
||||||
|
|
||||||
@@ -85,8 +86,8 @@ class RevPiOption(QtWidgets.QDialog, Ui_diag_options):
|
|||||||
self.cbb_reset_driver_action.currentIndex() != self.dc.get("reset_driver_action", 2) or
|
self.cbb_reset_driver_action.currentIndex() != self.dc.get("reset_driver_action", 2) or
|
||||||
# todo: self.dc.get("rtlevel", 2)
|
# todo: self.dc.get("rtlevel", 2)
|
||||||
|
|
||||||
int(self.cbx_plcslave.isChecked()) != self.dc.get("plcslave", 0) or
|
int(self.cbx_plcserver.isChecked()) != self.dc.get("plcserver", 0) or
|
||||||
self.acl_plcslave != self.dc.get("plcslaveacl", "") or
|
self.acl_plcserver != self.dc.get("plcserveracl", "") or
|
||||||
|
|
||||||
int(self.cbx_mqtt.isChecked()) != self.dc.get("mqtt", 0) or
|
int(self.cbx_mqtt.isChecked()) != self.dc.get("mqtt", 0) or
|
||||||
self._changesdone_mqtt() or
|
self._changesdone_mqtt() or
|
||||||
@@ -120,8 +121,8 @@ class RevPiOption(QtWidgets.QDialog, Ui_diag_options):
|
|||||||
self.cbx_zeroonerror.setChecked(bool(self.dc.get("zeroonerror", 0)))
|
self.cbx_zeroonerror.setChecked(bool(self.dc.get("zeroonerror", 0)))
|
||||||
self.txt_replace_io.setText(self.dc.get("replace_ios", ""))
|
self.txt_replace_io.setText(self.dc.get("replace_ios", ""))
|
||||||
self.cbb_reset_driver_action.setCurrentIndex(self.dc.get("reset_driver_action", 2))
|
self.cbb_reset_driver_action.setCurrentIndex(self.dc.get("reset_driver_action", 2))
|
||||||
self.cbx_plcslave.setChecked(bool(self.dc.get("plcslave", 0)))
|
self.cbx_plcserver.setChecked(bool(self.dc.get("plcserver", 0)))
|
||||||
self.acl_plcslave = self.dc.get("plcslaveacl", "")
|
self.acl_plcserver = self.dc.get("plcserveracl", "")
|
||||||
self.cbx_mqtt.setChecked(bool(self.dc.get("mqtt", 0)))
|
self.cbx_mqtt.setChecked(bool(self.dc.get("mqtt", 0)))
|
||||||
self.cbx_xmlrpc.setChecked(bool(self.dc.get("xmlrpc", 0)))
|
self.cbx_xmlrpc.setChecked(bool(self.dc.get("xmlrpc", 0)))
|
||||||
self.acl_xmlrpc = self.dc.get("xmlrpcacl", "")
|
self.acl_xmlrpc = self.dc.get("xmlrpcacl", "")
|
||||||
@@ -141,9 +142,44 @@ class RevPiOption(QtWidgets.QDialog, Ui_diag_options):
|
|||||||
if key in self.dc:
|
if key in self.dc:
|
||||||
self._dict_mqttsettings[key] = self.dc[key]
|
self._dict_mqttsettings[key] = self.dc[key]
|
||||||
|
|
||||||
|
def _translate_wrong_names(self) -> dict:
|
||||||
|
"""
|
||||||
|
Translate settings values of revpipyload < 0.10.0.
|
||||||
|
|
||||||
|
With RevPiPyLoad 0.10.0 we replaced the words master-slave with
|
||||||
|
client-server. Unfortunately we cannot expect that everyone will be
|
||||||
|
able to switch to the new version of RevPiPyLoad immediately.
|
||||||
|
Therefore, for a few versions of this software, we need to do a
|
||||||
|
translation of the values.
|
||||||
|
|
||||||
|
This function will translate the self.dc always to the new values and
|
||||||
|
return a copy of it with new or old values, depending on previous
|
||||||
|
detections.
|
||||||
|
|
||||||
|
:return: Settings with wrong values, if detected in previous calls
|
||||||
|
"""
|
||||||
|
name_mappings = (
|
||||||
|
("plcslave", "plcserver"),
|
||||||
|
("plcslaveacl", "plcserveracl"),
|
||||||
|
)
|
||||||
|
for wrong, right in name_mappings:
|
||||||
|
if wrong in self.dc:
|
||||||
|
self.wrong_names = True
|
||||||
|
self.dc[right] = self.dc[wrong]
|
||||||
|
del self.dc[wrong]
|
||||||
|
|
||||||
|
translated_settings = self.dc.copy()
|
||||||
|
if self.wrong_names:
|
||||||
|
for wrong, right in name_mappings:
|
||||||
|
if right in translated_settings:
|
||||||
|
translated_settings[wrong] = self.dc[right]
|
||||||
|
del translated_settings[right]
|
||||||
|
|
||||||
|
return translated_settings
|
||||||
|
|
||||||
def accept(self) -> None:
|
def accept(self) -> None:
|
||||||
if not self._changesdone():
|
if not self._changesdone():
|
||||||
super(RevPiOption, self).accept()
|
super().accept()
|
||||||
return
|
return
|
||||||
|
|
||||||
ask = QtWidgets.QMessageBox.question(
|
ask = QtWidgets.QMessageBox.question(
|
||||||
@@ -164,9 +200,9 @@ class RevPiOption(QtWidgets.QDialog, Ui_diag_options):
|
|||||||
self.dc["zeroonerror"] = int(self.cbx_zeroonerror.isChecked())
|
self.dc["zeroonerror"] = int(self.cbx_zeroonerror.isChecked())
|
||||||
self.dc["replace_ios"] = self.txt_replace_io.text()
|
self.dc["replace_ios"] = self.txt_replace_io.text()
|
||||||
|
|
||||||
# PLCSlave Settings
|
# PLCServer Settings
|
||||||
self.dc["plcslave"] = int(self.cbx_plcslave.isChecked())
|
self.dc["plcserver"] = int(self.cbx_plcserver.isChecked())
|
||||||
self.dc["plcslaveacl"] = self.acl_plcslave
|
self.dc["plcserveracl"] = self.acl_plcserver
|
||||||
|
|
||||||
# MQTT Settings
|
# MQTT Settings
|
||||||
self.dc["mqtt"] = int(self.cbx_mqtt.isChecked())
|
self.dc["mqtt"] = int(self.cbx_mqtt.isChecked())
|
||||||
@@ -186,12 +222,12 @@ class RevPiOption(QtWidgets.QDialog, Ui_diag_options):
|
|||||||
self.dc["xmlrpcacl"] = self.acl_xmlrpc
|
self.dc["xmlrpcacl"] = self.acl_xmlrpc
|
||||||
|
|
||||||
saved = helper.cm.call_remote_function(
|
saved = helper.cm.call_remote_function(
|
||||||
"set_config", self.dc, ask,
|
"set_config", self._translate_wrong_names(), ask,
|
||||||
default_value=False
|
default_value=False
|
||||||
)
|
)
|
||||||
|
|
||||||
if saved:
|
if saved:
|
||||||
super(RevPiOption, self).accept()
|
super().accept()
|
||||||
else:
|
else:
|
||||||
QtWidgets.QMessageBox.critical(
|
QtWidgets.QMessageBox.critical(
|
||||||
self, self.tr("Error"), self.tr(
|
self, self.tr("Error"), self.tr(
|
||||||
@@ -223,14 +259,20 @@ class RevPiOption(QtWidgets.QDialog, Ui_diag_options):
|
|||||||
if len(self.dc) == 0:
|
if len(self.dc) == 0:
|
||||||
return QtWidgets.QDialog.Rejected
|
return QtWidgets.QDialog.Rejected
|
||||||
|
|
||||||
|
self.wrong_names = False
|
||||||
|
self._translate_wrong_names()
|
||||||
|
|
||||||
self._load_settings()
|
self._load_settings()
|
||||||
self._apply_acl()
|
self._apply_acl()
|
||||||
|
|
||||||
running = helper.cm.call_remote_function("plcslaverunning", default_value=False)
|
running = helper.cm.call_remote_function(
|
||||||
self.lbl_slave_status.setText(
|
"plcslaverunning" if self.wrong_names else "plcserverrunning",
|
||||||
|
default_value=False
|
||||||
|
)
|
||||||
|
self.lbl_server_status.setText(
|
||||||
self.tr("running") if running else self.tr("stopped")
|
self.tr("running") if running else self.tr("stopped")
|
||||||
)
|
)
|
||||||
self.lbl_slave_status.setStyleSheet(
|
self.lbl_server_status.setStyleSheet(
|
||||||
"color: green" if running else "color: red"
|
"color: green" if running else "color: red"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -253,12 +295,12 @@ class RevPiOption(QtWidgets.QDialog, Ui_diag_options):
|
|||||||
"color: green" if running else "color: red"
|
"color: green" if running else "color: red"
|
||||||
)
|
)
|
||||||
|
|
||||||
return super(RevPiOption, self).exec()
|
return super().exec()
|
||||||
|
|
||||||
def reject(self) -> None:
|
def reject(self) -> None:
|
||||||
"""Reject all sub windows and reload settings."""
|
"""Reject all sub windows and reload settings."""
|
||||||
self._load_settings()
|
self._load_settings()
|
||||||
super(RevPiOption, self).reject()
|
super().reject()
|
||||||
|
|
||||||
@QtCore.pyqtSlot(int)
|
@QtCore.pyqtSlot(int)
|
||||||
def on_cbb_replace_io_currentIndexChanged(self, index: int):
|
def on_cbb_replace_io_currentIndexChanged(self, index: int):
|
||||||
@@ -275,15 +317,15 @@ class RevPiOption(QtWidgets.QDialog, Ui_diag_options):
|
|||||||
self.txt_replace_io.setEnabled(index == 3)
|
self.txt_replace_io.setEnabled(index == 3)
|
||||||
|
|
||||||
@QtCore.pyqtSlot()
|
@QtCore.pyqtSlot()
|
||||||
def on_btn_aclplcslave_clicked(self):
|
def on_btn_aclplcserver_clicked(self):
|
||||||
"""Start ACL manager to edit ACL entries."""
|
"""Start ACL manager to edit ACL entries."""
|
||||||
self.diag_aclmanager.setup_acl_manager(self.acl_plcslave, {
|
self.diag_aclmanager.setup_acl_manager(self.acl_plcserver, {
|
||||||
0: self.tr("read only"),
|
0: self.tr("read only"),
|
||||||
1: self.tr("read and write"),
|
1: self.tr("read and write"),
|
||||||
})
|
})
|
||||||
self.diag_aclmanager.read_only = helper.cm.xml_mode < 4
|
self.diag_aclmanager.read_only = helper.cm.xml_mode < 4
|
||||||
if self.diag_aclmanager.exec() == QtWidgets.QDialog.Accepted:
|
if self.diag_aclmanager.exec() == QtWidgets.QDialog.Accepted:
|
||||||
self.acl_plcslave = self.diag_aclmanager.get_acl()
|
self.acl_plcserver = self.diag_aclmanager.get_acl()
|
||||||
|
|
||||||
@QtCore.pyqtSlot()
|
@QtCore.pyqtSlot()
|
||||||
def on_btn_mqtt_clicked(self):
|
def on_btn_mqtt_clicked(self):
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""Saved connections of Revolution Pi devices."""
|
"""Saved connections of Revolution Pi devices."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
from enum import IntEnum
|
from enum import IntEnum
|
||||||
|
|
||||||
@@ -25,7 +25,7 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
"""Manage your saved connections."""
|
"""Manage your saved connections."""
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
super(RevPiPlcList, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
self.__default_port = 55123
|
self.__default_port = 55123
|
||||||
|
|
||||||
@@ -37,13 +37,30 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
self.lbl_port.setText(self.lbl_port.text().format(self.__default_port))
|
self.lbl_port.setText(self.lbl_port.text().format(self.__default_port))
|
||||||
self.sbx_port.setValue(self.__default_port)
|
self.sbx_port.setValue(self.__default_port)
|
||||||
|
|
||||||
|
# Dirty workaround to remove default button to prevent action on ENTER key, while user edit texts
|
||||||
|
self.__btn_dummy = QtWidgets.QPushButton(self)
|
||||||
|
self.__btn_dummy.setVisible(False)
|
||||||
|
self.__btn_dummy.setDefault(True)
|
||||||
|
|
||||||
|
def _load_cbb_folder(self):
|
||||||
|
"""Clean up all entries and reload existing ones from treeview."""
|
||||||
|
self.cbb_folder.blockSignals(True)
|
||||||
|
|
||||||
|
self.cbb_folder.clear()
|
||||||
|
self.cbb_folder.addItem("")
|
||||||
|
for i in range(self.tre_connections.topLevelItemCount()):
|
||||||
|
item = self.tre_connections.topLevelItem(i)
|
||||||
|
if item.type() != NodeType.DIR:
|
||||||
|
continue
|
||||||
|
self.cbb_folder.addItem(item.text(0))
|
||||||
|
|
||||||
|
self.cbb_folder.blockSignals(False)
|
||||||
|
|
||||||
def _load_settings(self):
|
def _load_settings(self):
|
||||||
"""Load values to GUI widgets."""
|
"""Load values to GUI widgets."""
|
||||||
pi.logger.debug("RevPiPlcList._load_settings")
|
pi.logger.debug("RevPiPlcList._load_settings")
|
||||||
|
|
||||||
self.tre_connections.clear()
|
self.tre_connections.clear()
|
||||||
self.cbb_folder.clear()
|
|
||||||
self.cbb_folder.addItem("")
|
|
||||||
|
|
||||||
# Get length of array and close it, the RevPiSettings-class need it
|
# Get length of array and close it, the RevPiSettings-class need it
|
||||||
count_settings = helper.settings.beginReadArray("connections")
|
count_settings = helper.settings.beginReadArray("connections")
|
||||||
@@ -67,7 +84,6 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
sub_folder.setIcon(0, QtGui.QIcon(":/main/ico/folder.ico"))
|
sub_folder.setIcon(0, QtGui.QIcon(":/main/ico/folder.ico"))
|
||||||
sub_folder.setText(0, folder)
|
sub_folder.setText(0, folder)
|
||||||
self.tre_connections.addTopLevelItem(sub_folder)
|
self.tre_connections.addTopLevelItem(sub_folder)
|
||||||
self.cbb_folder.addItem(folder)
|
|
||||||
|
|
||||||
sub_folder.addChild(con_item)
|
sub_folder.addChild(con_item)
|
||||||
else:
|
else:
|
||||||
@@ -76,7 +92,6 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
self.tre_connections.expandAll()
|
self.tre_connections.expandAll()
|
||||||
self.changes = False
|
self.changes = False
|
||||||
|
|
||||||
if self.tre_connections.topLevelItemCount() == 0:
|
|
||||||
self._edit_state()
|
self._edit_state()
|
||||||
|
|
||||||
def accept(self) -> None:
|
def accept(self) -> None:
|
||||||
@@ -109,7 +124,7 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
revpi_settings.save_settings()
|
revpi_settings.save_settings()
|
||||||
|
|
||||||
self.changes = False
|
self.changes = False
|
||||||
super(RevPiPlcList, self).accept()
|
super().accept()
|
||||||
|
|
||||||
def closeEvent(self, a0: QtGui.QCloseEvent) -> None:
|
def closeEvent(self, a0: QtGui.QCloseEvent) -> None:
|
||||||
pi.logger.debug("RevPiPlcList.closeEvent")
|
pi.logger.debug("RevPiPlcList.closeEvent")
|
||||||
@@ -127,7 +142,7 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
|
|
||||||
def exec(self) -> int:
|
def exec(self) -> int:
|
||||||
self._load_settings()
|
self._load_settings()
|
||||||
return super(RevPiPlcList, self).exec()
|
return super().exec()
|
||||||
|
|
||||||
def exec_with_presets(self, presets: RevPiSettings) -> int:
|
def exec_with_presets(self, presets: RevPiSettings) -> int:
|
||||||
"""
|
"""
|
||||||
@@ -138,7 +153,7 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
"""
|
"""
|
||||||
self._load_settings()
|
self._load_settings()
|
||||||
self.on_btn_add_clicked(presets)
|
self.on_btn_add_clicked(presets)
|
||||||
return super(RevPiPlcList, self).exec()
|
return super().exec()
|
||||||
|
|
||||||
@QtCore.pyqtSlot(QtWidgets.QAbstractButton)
|
@QtCore.pyqtSlot(QtWidgets.QAbstractButton)
|
||||||
def on_btn_box_clicked(self, button: QtWidgets.QAbstractButton):
|
def on_btn_box_clicked(self, button: QtWidgets.QAbstractButton):
|
||||||
@@ -150,6 +165,8 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
|
|
||||||
def _edit_state(self):
|
def _edit_state(self):
|
||||||
"""Set enabled status of all controls, depending on selected item."""
|
"""Set enabled status of all controls, depending on selected item."""
|
||||||
|
pi.logger.debug("RevPiPlcList._edit_state")
|
||||||
|
|
||||||
item = self.tre_connections.currentItem()
|
item = self.tre_connections.currentItem()
|
||||||
if item is None:
|
if item is None:
|
||||||
up_ok = False
|
up_ok = False
|
||||||
@@ -171,12 +188,16 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
|
|
||||||
self.btn_up.setEnabled(up_ok)
|
self.btn_up.setEnabled(up_ok)
|
||||||
self.btn_down.setEnabled(down_ok)
|
self.btn_down.setEnabled(down_ok)
|
||||||
self.btn_delete.setEnabled(con_item)
|
self.btn_delete.setEnabled(con_item or dir_item)
|
||||||
self.txt_name.setEnabled(con_item)
|
self.txt_name.setEnabled(con_item)
|
||||||
self.txt_address.setEnabled(con_item)
|
self.txt_address.setEnabled(con_item)
|
||||||
self.sbx_port.setEnabled(con_item)
|
self.sbx_port.setEnabled(con_item)
|
||||||
self.sbx_timeout.setEnabled(con_item)
|
self.sbx_timeout.setEnabled(con_item)
|
||||||
self.cbb_folder.setEnabled(con_item or dir_item)
|
self.cbb_folder.setEnabled(con_item or dir_item)
|
||||||
|
self.cbb_folder.setEditable(dir_item)
|
||||||
|
if self.cbb_folder.isEditable():
|
||||||
|
# Disable auto complete, this would override a new typed name with existing one
|
||||||
|
self.cbb_folder.setCompleter(None)
|
||||||
|
|
||||||
self.cbx_ssh_use_tunnel.setEnabled(con_item)
|
self.cbx_ssh_use_tunnel.setEnabled(con_item)
|
||||||
self.sbx_ssh_port.setEnabled(con_item)
|
self.sbx_ssh_port.setEnabled(con_item)
|
||||||
@@ -203,22 +224,25 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
if 0 <= new_index < dir_item.childCount():
|
if 0 <= new_index < dir_item.childCount():
|
||||||
item = dir_item.takeChild(index)
|
item = dir_item.takeChild(index)
|
||||||
dir_item.insertChild(new_index, item)
|
dir_item.insertChild(new_index, item)
|
||||||
self.tre_connections.expandItem(dir_item)
|
|
||||||
else:
|
else:
|
||||||
index = self.tre_connections.indexOfTopLevelItem(item)
|
index = self.tre_connections.indexOfTopLevelItem(item)
|
||||||
new_index = index + count
|
new_index = index + count
|
||||||
if 0 <= index < self.tre_connections.topLevelItemCount():
|
if 0 <= index < self.tre_connections.topLevelItemCount():
|
||||||
item = self.tre_connections.takeTopLevelItem(index)
|
item = self.tre_connections.takeTopLevelItem(index)
|
||||||
self.tre_connections.insertTopLevelItem(new_index, item)
|
self.tre_connections.insertTopLevelItem(new_index, item)
|
||||||
|
if item.type() == NodeType.DIR:
|
||||||
|
# Expand a moved dir node, it would be collapsed after move
|
||||||
|
self.tre_connections.expandItem(item)
|
||||||
|
|
||||||
self.tre_connections.setCurrentItem(item)
|
self.tre_connections.setCurrentItem(item)
|
||||||
self._edit_state()
|
|
||||||
|
|
||||||
@QtCore.pyqtSlot(QtWidgets.QTreeWidgetItem, QtWidgets.QTreeWidgetItem)
|
@QtCore.pyqtSlot(QtWidgets.QTreeWidgetItem, QtWidgets.QTreeWidgetItem)
|
||||||
def on_tre_connections_currentItemChanged(
|
def on_tre_connections_currentItemChanged(
|
||||||
self, current: QtWidgets.QTreeWidgetItem, previous: QtWidgets.QTreeWidgetItem):
|
self, current: QtWidgets.QTreeWidgetItem, previous: QtWidgets.QTreeWidgetItem):
|
||||||
|
|
||||||
self._edit_state()
|
self._edit_state()
|
||||||
|
self._load_cbb_folder()
|
||||||
|
|
||||||
if current and current.type() == NodeType.CON:
|
if current and current.type() == NodeType.CON:
|
||||||
self.__current_item = current
|
self.__current_item = current
|
||||||
|
|
||||||
@@ -255,9 +279,9 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
@QtCore.pyqtSlot()
|
@QtCore.pyqtSlot()
|
||||||
def on_btn_delete_clicked(self):
|
def on_btn_delete_clicked(self):
|
||||||
"""Remove selected entry."""
|
"""Remove selected entry."""
|
||||||
item = self.tre_connections.currentItem()
|
|
||||||
if item and item.type() == NodeType.CON:
|
|
||||||
|
|
||||||
|
def remove_item(item: QtWidgets.QTreeWidgetItem):
|
||||||
|
"""Remove CON item and save keyring actions for save action."""
|
||||||
revpi_settings = item.data(0, WidgetData.revpi_settings) # type: RevPiSettings
|
revpi_settings = item.data(0, WidgetData.revpi_settings) # type: RevPiSettings
|
||||||
if revpi_settings.ssh_saved_password:
|
if revpi_settings.ssh_saved_password:
|
||||||
# Cleans up keyring in save function
|
# Cleans up keyring in save function
|
||||||
@@ -270,7 +294,27 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
index = self.tre_connections.indexOfTopLevelItem(item)
|
index = self.tre_connections.indexOfTopLevelItem(item)
|
||||||
self.tre_connections.takeTopLevelItem(index)
|
self.tre_connections.takeTopLevelItem(index)
|
||||||
|
|
||||||
self._edit_state()
|
item_to_remove = self.tre_connections.currentItem()
|
||||||
|
|
||||||
|
if item_to_remove and item_to_remove.type() == NodeType.DIR:
|
||||||
|
if item_to_remove.childCount():
|
||||||
|
rc = QtWidgets.QMessageBox.question(
|
||||||
|
self, self.tr("Question"), self.tr(
|
||||||
|
"If you remote this folder, all containing elements will be removed, too. \n\n"
|
||||||
|
"Do you want to delete folder and all elements?"
|
||||||
|
),
|
||||||
|
)
|
||||||
|
if rc != QtWidgets.QMessageBox.Yes:
|
||||||
|
return
|
||||||
|
|
||||||
|
while item_to_remove.childCount() > 0:
|
||||||
|
remove_item(item_to_remove.child(0))
|
||||||
|
|
||||||
|
item_index = self.tre_connections.indexOfTopLevelItem(item_to_remove)
|
||||||
|
self.tre_connections.takeTopLevelItem(item_index)
|
||||||
|
|
||||||
|
elif item_to_remove and item_to_remove.type() == NodeType.CON:
|
||||||
|
remove_item(item_to_remove)
|
||||||
|
|
||||||
@QtCore.pyqtSlot()
|
@QtCore.pyqtSlot()
|
||||||
def on_btn_add_clicked(self, settings_preset: RevPiSettings = None):
|
def on_btn_add_clicked(self, settings_preset: RevPiSettings = None):
|
||||||
@@ -291,6 +335,20 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
self.txt_name.setFocus()
|
self.txt_name.setFocus()
|
||||||
self.txt_name.selectAll()
|
self.txt_name.selectAll()
|
||||||
|
|
||||||
|
@QtCore.pyqtSlot()
|
||||||
|
def on_btn_add_dir_clicked(self):
|
||||||
|
"""Add a new folder."""
|
||||||
|
folder_text = self.tr("New folder")
|
||||||
|
sub_folder = QtWidgets.QTreeWidgetItem(NodeType.DIR)
|
||||||
|
sub_folder.setIcon(0, QtGui.QIcon(":/main/ico/folder.ico"))
|
||||||
|
sub_folder.setText(0, folder_text)
|
||||||
|
|
||||||
|
self.tre_connections.addTopLevelItem(sub_folder)
|
||||||
|
self.tre_connections.setCurrentItem(sub_folder)
|
||||||
|
self.cbb_folder.setFocus()
|
||||||
|
|
||||||
|
self.changes = True
|
||||||
|
|
||||||
@QtCore.pyqtSlot(str)
|
@QtCore.pyqtSlot(str)
|
||||||
def on_txt_name_textEdited(self, text):
|
def on_txt_name_textEdited(self, text):
|
||||||
if self.__current_item.type() != NodeType.CON:
|
if self.__current_item.type() != NodeType.CON:
|
||||||
@@ -349,65 +407,47 @@ class RevPiPlcList(QtWidgets.QDialog, Ui_diag_connections):
|
|||||||
settings.ssh_user = text
|
settings.ssh_user = text
|
||||||
self.changes = True
|
self.changes = True
|
||||||
|
|
||||||
|
@QtCore.pyqtSlot(str)
|
||||||
|
def on_cbb_folder_currentIndexChanged(self, text: str):
|
||||||
|
pi.logger.debug("RevPiPlcList.on_cbb_folder_currentIndexChanged({0})".format(text))
|
||||||
|
|
||||||
|
if self.__current_item.type() == NodeType.CON:
|
||||||
|
new_dir_node = self._get_folder_item(text)
|
||||||
|
current_dir_node = self.__current_item.parent()
|
||||||
|
if current_dir_node == new_dir_node:
|
||||||
|
# No change required, both nodes are the same
|
||||||
|
return
|
||||||
|
|
||||||
|
change_item = self.__current_item
|
||||||
|
self.tre_connections.blockSignals(True)
|
||||||
|
self.changes = True
|
||||||
|
|
||||||
|
if current_dir_node:
|
||||||
|
# Move an element to other folder or root
|
||||||
|
index = current_dir_node.indexOfChild(change_item)
|
||||||
|
change_item = current_dir_node.takeChild(index)
|
||||||
|
|
||||||
|
else:
|
||||||
|
# Move a root element to a folder
|
||||||
|
index = self.tre_connections.indexOfTopLevelItem(change_item)
|
||||||
|
change_item = self.tre_connections.takeTopLevelItem(index)
|
||||||
|
|
||||||
|
if text == "":
|
||||||
|
self.tre_connections.addTopLevelItem(change_item)
|
||||||
|
|
||||||
|
else:
|
||||||
|
new_dir_node.addChild(change_item)
|
||||||
|
|
||||||
|
self.tre_connections.blockSignals(False)
|
||||||
|
self.tre_connections.setCurrentItem(change_item)
|
||||||
|
|
||||||
@QtCore.pyqtSlot(str)
|
@QtCore.pyqtSlot(str)
|
||||||
def on_cbb_folder_editTextChanged(self, text: str):
|
def on_cbb_folder_editTextChanged(self, text: str):
|
||||||
pi.logger.debug("RevPiPlcList.on_cbb_folder_editTextChanged({0})".format(text))
|
pi.logger.debug("RevPiPlcList.on_cbb_folder_editTextChanged({0})".format(text))
|
||||||
|
|
||||||
if self.__current_item.type() == NodeType.DIR:
|
if self.__current_item.type() == NodeType.DIR and self.__current_item.text(0) != text:
|
||||||
# We just have to rename the dir node
|
# We just have to rename the dir node
|
||||||
self.__current_item.setText(0, text)
|
self.__current_item.setText(0, text)
|
||||||
|
self.changes = True
|
||||||
elif self.__current_item.type() == NodeType.CON:
|
|
||||||
sub_folder = self._get_folder_item(text)
|
|
||||||
dir_node = self.__current_item.parent()
|
|
||||||
if dir_node:
|
|
||||||
if dir_node.text(0) == text:
|
|
||||||
# It is the same folder
|
|
||||||
return
|
|
||||||
|
|
||||||
if text != "" and dir_node.childCount() == 1 and not sub_folder:
|
|
||||||
# The folder hold just one item, so we can rename that
|
|
||||||
for i in range(self.cbb_folder.count()):
|
|
||||||
if self.cbb_folder.itemText(i) == dir_node.text(0):
|
|
||||||
self.cbb_folder.setItemText(i, text)
|
|
||||||
break
|
|
||||||
dir_node.setText(0, text)
|
|
||||||
return
|
|
||||||
|
|
||||||
index = dir_node.indexOfChild(self.__current_item)
|
|
||||||
self.__current_item = dir_node.takeChild(index)
|
|
||||||
|
|
||||||
elif text != "":
|
|
||||||
# Move root to folder
|
|
||||||
index = self.tre_connections.indexOfTopLevelItem(self.__current_item)
|
|
||||||
self.__current_item = self.tre_connections.takeTopLevelItem(index)
|
|
||||||
|
|
||||||
else:
|
|
||||||
# Root stays root
|
|
||||||
return
|
|
||||||
|
|
||||||
if text == "":
|
|
||||||
self.tre_connections.addTopLevelItem(self.__current_item)
|
|
||||||
|
|
||||||
else:
|
|
||||||
if sub_folder is None:
|
|
||||||
sub_folder = QtWidgets.QTreeWidgetItem(NodeType.DIR)
|
|
||||||
sub_folder.setIcon(0, QtGui.QIcon(":/main/ico/folder.ico"))
|
|
||||||
sub_folder.setText(0, text)
|
|
||||||
self.tre_connections.addTopLevelItem(sub_folder)
|
|
||||||
self.cbb_folder.addItem(text)
|
|
||||||
sub_folder.addChild(self.__current_item)
|
|
||||||
|
|
||||||
if dir_node and dir_node.childCount() == 0:
|
|
||||||
# Remove empty folders
|
|
||||||
for i in range(self.cbb_folder.count()):
|
|
||||||
if self.cbb_folder.itemText(i) == dir_node.text(0):
|
|
||||||
self.cbb_folder.removeItem(i)
|
|
||||||
break
|
|
||||||
index = self.tre_connections.indexOfTopLevelItem(dir_node)
|
|
||||||
self.tre_connections.takeTopLevelItem(index)
|
|
||||||
|
|
||||||
self.tre_connections.setCurrentItem(self.__current_item)
|
|
||||||
self.cbb_folder.setFocus()
|
|
||||||
|
|
||||||
# endregion # # # # #
|
# endregion # # # # #
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""Revolution Pi PLC program configuration."""
|
"""Revolution Pi PLC program configuration."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
import gzip
|
import gzip
|
||||||
import os
|
import os
|
||||||
@@ -23,7 +23,7 @@ class RevPiProgram(QtWidgets.QDialog, Ui_diag_program):
|
|||||||
"""Program options of RevPiPyLoad."""
|
"""Program options of RevPiPyLoad."""
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
super(RevPiProgram, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
self.setFixedSize(self.size())
|
self.setFixedSize(self.size())
|
||||||
|
|
||||||
@@ -103,7 +103,7 @@ class RevPiProgram(QtWidgets.QDialog, Ui_diag_program):
|
|||||||
def accept(self) -> None:
|
def accept(self) -> None:
|
||||||
# todo: After upload ask for restart pcl program?
|
# todo: After upload ask for restart pcl program?
|
||||||
if not self._changesdone():
|
if not self._changesdone():
|
||||||
super(RevPiProgram, self).accept()
|
super().accept()
|
||||||
return
|
return
|
||||||
|
|
||||||
if self.cbb_plcprogram.currentText() == "":
|
if self.cbb_plcprogram.currentText() == "":
|
||||||
@@ -138,7 +138,7 @@ class RevPiProgram(QtWidgets.QDialog, Ui_diag_program):
|
|||||||
)
|
)
|
||||||
|
|
||||||
if saved:
|
if saved:
|
||||||
super(RevPiProgram, self).accept()
|
super().accept()
|
||||||
else:
|
else:
|
||||||
QtWidgets.QMessageBox.critical(
|
QtWidgets.QMessageBox.critical(
|
||||||
self, self.tr("Error"), self.tr(
|
self, self.tr("Error"), self.tr(
|
||||||
@@ -167,19 +167,19 @@ class RevPiProgram(QtWidgets.QDialog, Ui_diag_program):
|
|||||||
return QtWidgets.QDialog.Rejected
|
return QtWidgets.QDialog.Rejected
|
||||||
|
|
||||||
self.dc = helper.cm.call_remote_function("get_config", default_value={})
|
self.dc = helper.cm.call_remote_function("get_config", default_value={})
|
||||||
self.lst_files = helper.cm.call_remote_function("get_filelist", default_value=[])
|
self.lst_files = helper.cm.call_remote_function("get_filelist", default_value=None)
|
||||||
if len(self.dc) == 0 or len(self.lst_files) == 0:
|
if len(self.dc) == 0 or self.lst_files is None:
|
||||||
return QtWidgets.QDialog.Rejected
|
return QtWidgets.QDialog.Rejected
|
||||||
|
|
||||||
self._load_settings()
|
self._load_settings()
|
||||||
self._apply_acl()
|
self._apply_acl()
|
||||||
|
|
||||||
return super(RevPiProgram, self).exec()
|
return super().exec()
|
||||||
|
|
||||||
def reject(self) -> None:
|
def reject(self) -> None:
|
||||||
"""Reject all sub windows and reload settings."""
|
"""Reject all sub windows and reload settings."""
|
||||||
self._load_settings()
|
self._load_settings()
|
||||||
super(RevPiProgram, self).reject()
|
super().reject()
|
||||||
|
|
||||||
# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
|
# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
|
||||||
# region # REGION: PLC program
|
# region # REGION: PLC program
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""Simulator for piControl."""
|
"""Simulator for piControl."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
from os import W_OK, access
|
from os import W_OK, access
|
||||||
from os.path import basename, dirname, exists, join
|
from os.path import basename, dirname, exists, join
|
||||||
@@ -21,7 +21,7 @@ class Simulator(QtWidgets.QDialog, Ui_diag_simulator):
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
super(Simulator, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
self.clean_procimg = False
|
self.clean_procimg = False
|
||||||
self.max_items = 5
|
self.max_items = 5
|
||||||
@@ -69,7 +69,7 @@ class Simulator(QtWidgets.QDialog, Ui_diag_simulator):
|
|||||||
|
|
||||||
self.clean_procimg = self.sender() is self.btn_start_empty
|
self.clean_procimg = self.sender() is self.btn_start_empty
|
||||||
|
|
||||||
super(Simulator, self).accept()
|
super().accept()
|
||||||
|
|
||||||
def closeEvent(self, a0: QtGui.QCloseEvent) -> None:
|
def closeEvent(self, a0: QtGui.QCloseEvent) -> None:
|
||||||
self._save_gui()
|
self._save_gui()
|
||||||
|
|||||||
@@ -2,4 +2,4 @@
|
|||||||
"""Package of ssh tunnel connections."""
|
"""Package of ssh tunnel connections."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|||||||
@@ -7,17 +7,21 @@ GitHub https://github.com/paramiko/paramiko/blob/main/demos/forward.py
|
|||||||
"""
|
"""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
import select
|
import select
|
||||||
|
from logging import getLogger
|
||||||
from socketserver import BaseRequestHandler, ThreadingTCPServer
|
from socketserver import BaseRequestHandler, ThreadingTCPServer
|
||||||
from threading import Thread
|
from threading import Thread
|
||||||
|
from typing import Tuple, Union
|
||||||
|
|
||||||
from paramiko.client import SSHClient, WarningPolicy
|
from paramiko.client import MissingHostKeyPolicy, SSHClient
|
||||||
from paramiko.rsakey import RSAKey
|
from paramiko.rsakey import RSAKey
|
||||||
from paramiko.ssh_exception import PasswordRequiredException
|
from paramiko.ssh_exception import PasswordRequiredException
|
||||||
from paramiko.transport import Transport
|
from paramiko.transport import Transport
|
||||||
|
|
||||||
|
log = getLogger("ssh_tunneling")
|
||||||
|
|
||||||
|
|
||||||
class ForwardServer(ThreadingTCPServer):
|
class ForwardServer(ThreadingTCPServer):
|
||||||
daemon_threads = True
|
daemon_threads = True
|
||||||
@@ -33,10 +37,14 @@ class Handler(BaseRequestHandler):
|
|||||||
self.request.getpeername(),
|
self.request.getpeername(),
|
||||||
)
|
)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
log.error(e)
|
||||||
return
|
return
|
||||||
if chan is None:
|
if chan is None:
|
||||||
|
log.error("Could not create a ssh channel")
|
||||||
return
|
return
|
||||||
|
|
||||||
|
log.info("Starting tunnel exchange loop")
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
r, w, x = select.select([self.request, chan], [], [], 5.0)
|
r, w, x = select.select([self.request, chan], [], [], 5.0)
|
||||||
if self.request in r:
|
if self.request in r:
|
||||||
@@ -50,6 +58,8 @@ class Handler(BaseRequestHandler):
|
|||||||
break
|
break
|
||||||
self.request.send(data)
|
self.request.send(data)
|
||||||
|
|
||||||
|
log.info("Stopped tunnel exchange loop")
|
||||||
|
|
||||||
chan.close()
|
chan.close()
|
||||||
self.request.close()
|
self.request.close()
|
||||||
|
|
||||||
@@ -71,8 +81,7 @@ class SSHLocalTunnel:
|
|||||||
self._th_server = Thread()
|
self._th_server = Thread()
|
||||||
|
|
||||||
self._ssh_client = SSHClient()
|
self._ssh_client = SSHClient()
|
||||||
self._ssh_client.load_system_host_keys()
|
self._ssh_client.set_missing_host_key_policy(MissingHostKeyPolicy())
|
||||||
self._ssh_client.set_missing_host_key_policy(WarningPolicy())
|
|
||||||
|
|
||||||
self._ssh_transport = None # type: Transport
|
self._ssh_transport = None # type: Transport
|
||||||
self._forward_server = None # type: ThreadingTCPServer
|
self._forward_server = None # type: ThreadingTCPServer
|
||||||
@@ -163,6 +172,30 @@ class SSHLocalTunnel:
|
|||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
def send_cmd(self, cmd: str, timeout: float = None) -> Union[Tuple[str, str], Tuple[None, None]]:
|
||||||
|
"""
|
||||||
|
Send simple command to ssh host.
|
||||||
|
|
||||||
|
The output of stdout and stderr is returned as a tuple of two elements.
|
||||||
|
This elements could be None, in case of an internal error.
|
||||||
|
|
||||||
|
:param cmd: Shell command to execute on remote host
|
||||||
|
:param timeout: Timeout for execution
|
||||||
|
:return: Tuple with stdout and stderr
|
||||||
|
"""
|
||||||
|
if not self._th_server.is_alive():
|
||||||
|
raise RuntimeError("Not connected")
|
||||||
|
|
||||||
|
try:
|
||||||
|
_, stdout, stderr = self._ssh_client.exec_command(cmd, 1024, timeout)
|
||||||
|
buffer_out = stdout.read()
|
||||||
|
buffer_err = stderr.read()
|
||||||
|
|
||||||
|
return buffer_out.decode(), buffer_err.decode()
|
||||||
|
except Exception as e:
|
||||||
|
log.error(e)
|
||||||
|
return None, None
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def connected(self):
|
def connected(self):
|
||||||
"""Check connection state of ssh tunnel."""
|
"""Check connection state of ssh tunnel."""
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"""Authentication dialog for SSH."""
|
"""Authentication dialog for SSH."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|
||||||
from logging import getLogger
|
from logging import getLogger
|
||||||
|
|
||||||
@@ -31,7 +31,7 @@ class SSHAuth(QtWidgets.QDialog, Ui_diag_sshauth):
|
|||||||
"""
|
"""
|
||||||
log.debug("SSHAuth.__init__")
|
log.debug("SSHAuth.__init__")
|
||||||
|
|
||||||
super(SSHAuth, self).__init__(parent)
|
super().__init__(parent)
|
||||||
self.setupUi(self)
|
self.setupUi(self)
|
||||||
|
|
||||||
self._in_keyring = False
|
self._in_keyring = False
|
||||||
|
|||||||
@@ -2,4 +2,4 @@
|
|||||||
"""Package of compiled Qt ui files."""
|
"""Package of compiled Qt ui files."""
|
||||||
__author__ = "Sven Sager"
|
__author__ = "Sven Sager"
|
||||||
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
__copyright__ = "Copyright (C) 2023 Sven Sager"
|
||||||
__license__ = "GPLv3"
|
__license__ = "GPLv2"
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ from PyQt5 import QtCore, QtGui, QtWidgets
|
|||||||
class Ui_diag_backgroundworker(object):
|
class Ui_diag_backgroundworker(object):
|
||||||
def setupUi(self, diag_backgroundworker):
|
def setupUi(self, diag_backgroundworker):
|
||||||
diag_backgroundworker.setObjectName("diag_backgroundworker")
|
diag_backgroundworker.setObjectName("diag_backgroundworker")
|
||||||
diag_backgroundworker.resize(418, 97)
|
diag_backgroundworker.resize(424, 104)
|
||||||
diag_backgroundworker.setModal(True)
|
diag_backgroundworker.setModal(True)
|
||||||
self.verticalLayout = QtWidgets.QVBoxLayout(diag_backgroundworker)
|
self.verticalLayout = QtWidgets.QVBoxLayout(diag_backgroundworker)
|
||||||
self.verticalLayout.setObjectName("verticalLayout")
|
self.verticalLayout.setObjectName("verticalLayout")
|
||||||
@@ -37,8 +37,7 @@ class Ui_diag_backgroundworker(object):
|
|||||||
QtCore.QMetaObject.connectSlotsByName(diag_backgroundworker)
|
QtCore.QMetaObject.connectSlotsByName(diag_backgroundworker)
|
||||||
|
|
||||||
def retranslateUi(self, diag_backgroundworker):
|
def retranslateUi(self, diag_backgroundworker):
|
||||||
_translate = QtCore.QCoreApplication.translate
|
pass
|
||||||
diag_backgroundworker.setWindowTitle(_translate("diag_backgroundworker", "File transfer..."))
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -45,10 +45,6 @@ class Ui_diag_revpiinfo(object):
|
|||||||
self.lbl_version_pyload.setAlignment(QtCore.Qt.AlignCenter)
|
self.lbl_version_pyload.setAlignment(QtCore.Qt.AlignCenter)
|
||||||
self.lbl_version_pyload.setObjectName("lbl_version_pyload")
|
self.lbl_version_pyload.setObjectName("lbl_version_pyload")
|
||||||
self.gridLayout.addWidget(self.lbl_version_pyload, 3, 1, 1, 1)
|
self.gridLayout.addWidget(self.lbl_version_pyload, 3, 1, 1, 1)
|
||||||
self.lbl_link = QtWidgets.QLabel(diag_revpiinfo)
|
|
||||||
self.lbl_link.setOpenExternalLinks(True)
|
|
||||||
self.lbl_link.setObjectName("lbl_link")
|
|
||||||
self.gridLayout.addWidget(self.lbl_link, 6, 0, 1, 1)
|
|
||||||
self.btn_box = QtWidgets.QDialogButtonBox(diag_revpiinfo)
|
self.btn_box = QtWidgets.QDialogButtonBox(diag_revpiinfo)
|
||||||
self.btn_box.setOrientation(QtCore.Qt.Horizontal)
|
self.btn_box.setOrientation(QtCore.Qt.Horizontal)
|
||||||
self.btn_box.setStandardButtons(QtWidgets.QDialogButtonBox.Ok)
|
self.btn_box.setStandardButtons(QtWidgets.QDialogButtonBox.Ok)
|
||||||
@@ -78,6 +74,11 @@ class Ui_diag_revpiinfo(object):
|
|||||||
self.gridLayout.addWidget(self.lbl_info, 4, 0, 1, 2)
|
self.gridLayout.addWidget(self.lbl_info, 4, 0, 1, 2)
|
||||||
spacerItem = QtWidgets.QSpacerItem(20, 40, QtWidgets.QSizePolicy.Minimum, QtWidgets.QSizePolicy.Expanding)
|
spacerItem = QtWidgets.QSpacerItem(20, 40, QtWidgets.QSizePolicy.Minimum, QtWidgets.QSizePolicy.Expanding)
|
||||||
self.gridLayout.addItem(spacerItem, 5, 0, 1, 1)
|
self.gridLayout.addItem(spacerItem, 5, 0, 1, 1)
|
||||||
|
self.lbl_link = QtWidgets.QLabel(diag_revpiinfo)
|
||||||
|
self.lbl_link.setText("<html><head/><body><p><a href=\"https://revpimodio.org/\"><span style=\" text-decoration: underline; color:#0000ff;\">https://revpimodio.org/</span></a></p></body></html>")
|
||||||
|
self.lbl_link.setOpenExternalLinks(True)
|
||||||
|
self.lbl_link.setObjectName("lbl_link")
|
||||||
|
self.gridLayout.addWidget(self.lbl_link, 6, 0, 1, 2)
|
||||||
|
|
||||||
self.retranslateUi(diag_revpiinfo)
|
self.retranslateUi(diag_revpiinfo)
|
||||||
self.btn_box.accepted.connect(diag_revpiinfo.accept) # type: ignore
|
self.btn_box.accepted.connect(diag_revpiinfo.accept) # type: ignore
|
||||||
@@ -89,13 +90,12 @@ class Ui_diag_revpiinfo(object):
|
|||||||
diag_revpiinfo.setWindowTitle(_translate("diag_revpiinfo", "Program information"))
|
diag_revpiinfo.setWindowTitle(_translate("diag_revpiinfo", "Program information"))
|
||||||
self.lbl_head.setText(_translate("diag_revpiinfo", "RevPi Python PLC - Commander"))
|
self.lbl_head.setText(_translate("diag_revpiinfo", "RevPi Python PLC - Commander"))
|
||||||
self.lbl_lbl_version_pyload.setText(_translate("diag_revpiinfo", "RevPiPyLoad version on RevPi:"))
|
self.lbl_lbl_version_pyload.setText(_translate("diag_revpiinfo", "RevPiPyLoad version on RevPi:"))
|
||||||
self.lbl_link.setText(_translate("diag_revpiinfo", "<html><head/><body><p><a href=\"https://revpimodio.org/\"><span style=\" text-decoration: underline; color:#0000ff;\">https://revpimodio.org/</span></a></p></body></html>"))
|
|
||||||
self.lbl_lbl_version_control.setText(_translate("diag_revpiinfo", "Version:"))
|
self.lbl_lbl_version_control.setText(_translate("diag_revpiinfo", "Version:"))
|
||||||
self.lbl_info.setText(_translate("diag_revpiinfo", "RevPiModIO, RevPiPyLoad and RevPiPyControl are community driven projects. They are all free and open source software.\n"
|
self.lbl_info.setText(_translate("diag_revpiinfo", "RevPiModIO, RevPiPyLoad and RevPiPyControl are community driven projects. They are all free and open source software.\n"
|
||||||
"All of them comes with ABSOLUTELY NO WARRANTY, to the extent permitted by\n"
|
"All of them comes with ABSOLUTELY NO WARRANTY, to the extent permitted by\n"
|
||||||
"applicable law.\n"
|
"applicable law.\n"
|
||||||
"\n"
|
"\n"
|
||||||
"(c) Sven Sager, License: GPLv3"))
|
"(c) Sven Sager, License: GPLv2"))
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|||||||
@@ -74,23 +74,23 @@ class Ui_diag_options(object):
|
|||||||
self.gb_server.setObjectName("gb_server")
|
self.gb_server.setObjectName("gb_server")
|
||||||
self.gridLayout_2 = QtWidgets.QGridLayout(self.gb_server)
|
self.gridLayout_2 = QtWidgets.QGridLayout(self.gb_server)
|
||||||
self.gridLayout_2.setObjectName("gridLayout_2")
|
self.gridLayout_2.setObjectName("gridLayout_2")
|
||||||
self.btn_aclplcslave = QtWidgets.QPushButton(self.gb_server)
|
self.btn_aclplcserver = QtWidgets.QPushButton(self.gb_server)
|
||||||
self.btn_aclplcslave.setObjectName("btn_aclplcslave")
|
self.btn_aclplcserver.setObjectName("btn_aclplcserver")
|
||||||
self.gridLayout_2.addWidget(self.btn_aclplcslave, 0, 1, 1, 1)
|
self.gridLayout_2.addWidget(self.btn_aclplcserver, 0, 1, 1, 1)
|
||||||
self.cbx_mqtt = QtWidgets.QCheckBox(self.gb_server)
|
self.cbx_mqtt = QtWidgets.QCheckBox(self.gb_server)
|
||||||
self.cbx_mqtt.setObjectName("cbx_mqtt")
|
self.cbx_mqtt.setObjectName("cbx_mqtt")
|
||||||
self.gridLayout_2.addWidget(self.cbx_mqtt, 2, 0, 1, 1)
|
self.gridLayout_2.addWidget(self.cbx_mqtt, 2, 0, 1, 1)
|
||||||
self.cbx_plcslave = QtWidgets.QCheckBox(self.gb_server)
|
self.cbx_plcserver = QtWidgets.QCheckBox(self.gb_server)
|
||||||
self.cbx_plcslave.setObjectName("cbx_plcslave")
|
self.cbx_plcserver.setObjectName("cbx_plcserver")
|
||||||
self.gridLayout_2.addWidget(self.cbx_plcslave, 0, 0, 1, 1)
|
self.gridLayout_2.addWidget(self.cbx_plcserver, 0, 0, 1, 1)
|
||||||
self.lbl_slave_status = QtWidgets.QLabel(self.gb_server)
|
self.lbl_server_status = QtWidgets.QLabel(self.gb_server)
|
||||||
self.lbl_slave_status.setAlignment(QtCore.Qt.AlignCenter)
|
self.lbl_server_status.setAlignment(QtCore.Qt.AlignCenter)
|
||||||
self.lbl_slave_status.setObjectName("lbl_slave_status")
|
self.lbl_server_status.setObjectName("lbl_server_status")
|
||||||
self.gridLayout_2.addWidget(self.lbl_slave_status, 1, 1, 1, 1)
|
self.gridLayout_2.addWidget(self.lbl_server_status, 1, 1, 1, 1)
|
||||||
self.lbl_lbl_slave_status = QtWidgets.QLabel(self.gb_server)
|
self.lbl_lbl_server_status = QtWidgets.QLabel(self.gb_server)
|
||||||
self.lbl_lbl_slave_status.setAlignment(QtCore.Qt.AlignRight|QtCore.Qt.AlignTrailing|QtCore.Qt.AlignVCenter)
|
self.lbl_lbl_server_status.setAlignment(QtCore.Qt.AlignRight|QtCore.Qt.AlignTrailing|QtCore.Qt.AlignVCenter)
|
||||||
self.lbl_lbl_slave_status.setObjectName("lbl_lbl_slave_status")
|
self.lbl_lbl_server_status.setObjectName("lbl_lbl_server_status")
|
||||||
self.gridLayout_2.addWidget(self.lbl_lbl_slave_status, 1, 0, 1, 1)
|
self.gridLayout_2.addWidget(self.lbl_lbl_server_status, 1, 0, 1, 1)
|
||||||
self.lbl_mqtt_status = QtWidgets.QLabel(self.gb_server)
|
self.lbl_mqtt_status = QtWidgets.QLabel(self.gb_server)
|
||||||
self.lbl_mqtt_status.setAlignment(QtCore.Qt.AlignCenter)
|
self.lbl_mqtt_status.setAlignment(QtCore.Qt.AlignCenter)
|
||||||
self.lbl_mqtt_status.setObjectName("lbl_mqtt_status")
|
self.lbl_mqtt_status.setObjectName("lbl_mqtt_status")
|
||||||
@@ -141,11 +141,11 @@ class Ui_diag_options(object):
|
|||||||
self.cbx_autoreload.setText(_translate("diag_options", "Restart PLC program after exit or crash"))
|
self.cbx_autoreload.setText(_translate("diag_options", "Restart PLC program after exit or crash"))
|
||||||
self.lbl_lbl_reset_driver_action.setText(_translate("diag_options", "PLC program behavior after piCtory driver reset clicked"))
|
self.lbl_lbl_reset_driver_action.setText(_translate("diag_options", "PLC program behavior after piCtory driver reset clicked"))
|
||||||
self.gb_server.setTitle(_translate("diag_options", "RevPiPyLoad server services"))
|
self.gb_server.setTitle(_translate("diag_options", "RevPiPyLoad server services"))
|
||||||
self.btn_aclplcslave.setText(_translate("diag_options", "Edit ACL"))
|
self.btn_aclplcserver.setText(_translate("diag_options", "Edit ACL"))
|
||||||
self.cbx_mqtt.setText(_translate("diag_options", "MQTT process image publisher"))
|
self.cbx_mqtt.setText(_translate("diag_options", "MQTT process image publisher"))
|
||||||
self.cbx_plcslave.setText(_translate("diag_options", "Start RevPi piControl server"))
|
self.cbx_plcserver.setText(_translate("diag_options", "Start RevPi piControl server"))
|
||||||
self.lbl_slave_status.setText(_translate("diag_options", "status"))
|
self.lbl_server_status.setText(_translate("diag_options", "status"))
|
||||||
self.lbl_lbl_slave_status.setText(_translate("diag_options", "piControl server is:"))
|
self.lbl_lbl_server_status.setText(_translate("diag_options", "piControl server is:"))
|
||||||
self.lbl_mqtt_status.setText(_translate("diag_options", "status"))
|
self.lbl_mqtt_status.setText(_translate("diag_options", "status"))
|
||||||
self.lbl_lbl_mqtt_status.setText(_translate("diag_options", "MQTT publish service is:"))
|
self.lbl_lbl_mqtt_status.setText(_translate("diag_options", "MQTT publish service is:"))
|
||||||
self.btn_mqtt.setText(_translate("diag_options", "Settings"))
|
self.btn_mqtt.setText(_translate("diag_options", "Settings"))
|
||||||
|
|||||||
@@ -67,6 +67,7 @@ class Ui_diag_connections(object):
|
|||||||
self.formLayout_2.setWidget(4, QtWidgets.QFormLayout.LabelRole, self.lbl_folder)
|
self.formLayout_2.setWidget(4, QtWidgets.QFormLayout.LabelRole, self.lbl_folder)
|
||||||
self.cbb_folder = QtWidgets.QComboBox(self.tab_connection)
|
self.cbb_folder = QtWidgets.QComboBox(self.tab_connection)
|
||||||
self.cbb_folder.setEditable(True)
|
self.cbb_folder.setEditable(True)
|
||||||
|
self.cbb_folder.setSizeAdjustPolicy(QtWidgets.QComboBox.AdjustToContents)
|
||||||
self.cbb_folder.setObjectName("cbb_folder")
|
self.cbb_folder.setObjectName("cbb_folder")
|
||||||
self.cbb_folder.addItem("")
|
self.cbb_folder.addItem("")
|
||||||
self.cbb_folder.setItemText(0, "")
|
self.cbb_folder.setItemText(0, "")
|
||||||
@@ -115,39 +116,41 @@ class Ui_diag_connections(object):
|
|||||||
spacerItem = QtWidgets.QSpacerItem(20, 40, QtWidgets.QSizePolicy.Minimum, QtWidgets.QSizePolicy.Expanding)
|
spacerItem = QtWidgets.QSpacerItem(20, 40, QtWidgets.QSizePolicy.Minimum, QtWidgets.QSizePolicy.Expanding)
|
||||||
self.vl_edit.addItem(spacerItem)
|
self.vl_edit.addItem(spacerItem)
|
||||||
self.btn_up = QtWidgets.QPushButton(diag_connections)
|
self.btn_up = QtWidgets.QPushButton(diag_connections)
|
||||||
self.btn_up.setText("")
|
|
||||||
icon = QtGui.QIcon()
|
icon = QtGui.QIcon()
|
||||||
icon.addPixmap(QtGui.QPixmap(":/action/ico/arrow-up.ico"), QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
icon.addPixmap(QtGui.QPixmap(":/action/ico/arrow-up.ico"), QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
||||||
self.btn_up.setIcon(icon)
|
self.btn_up.setIcon(icon)
|
||||||
self.btn_up.setObjectName("btn_up")
|
self.btn_up.setObjectName("btn_up")
|
||||||
self.vl_edit.addWidget(self.btn_up)
|
self.vl_edit.addWidget(self.btn_up)
|
||||||
self.btn_down = QtWidgets.QPushButton(diag_connections)
|
self.btn_down = QtWidgets.QPushButton(diag_connections)
|
||||||
self.btn_down.setText("")
|
|
||||||
icon1 = QtGui.QIcon()
|
icon1 = QtGui.QIcon()
|
||||||
icon1.addPixmap(QtGui.QPixmap(":/action/ico/arrow-down.ico"), QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
icon1.addPixmap(QtGui.QPixmap(":/action/ico/arrow-down.ico"), QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
||||||
self.btn_down.setIcon(icon1)
|
self.btn_down.setIcon(icon1)
|
||||||
self.btn_down.setObjectName("btn_down")
|
self.btn_down.setObjectName("btn_down")
|
||||||
self.vl_edit.addWidget(self.btn_down)
|
self.vl_edit.addWidget(self.btn_down)
|
||||||
self.btn_delete = QtWidgets.QPushButton(diag_connections)
|
self.btn_delete = QtWidgets.QPushButton(diag_connections)
|
||||||
self.btn_delete.setText("")
|
|
||||||
icon2 = QtGui.QIcon()
|
icon2 = QtGui.QIcon()
|
||||||
icon2.addPixmap(QtGui.QPixmap(":/action/ico/edit-delete.ico"), QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
icon2.addPixmap(QtGui.QPixmap(":/action/ico/edit-delete-6.ico"), QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
||||||
self.btn_delete.setIcon(icon2)
|
self.btn_delete.setIcon(icon2)
|
||||||
self.btn_delete.setObjectName("btn_delete")
|
self.btn_delete.setObjectName("btn_delete")
|
||||||
self.vl_edit.addWidget(self.btn_delete)
|
self.vl_edit.addWidget(self.btn_delete)
|
||||||
self.btn_add = QtWidgets.QPushButton(diag_connections)
|
self.btn_add_dir = QtWidgets.QPushButton(diag_connections)
|
||||||
self.btn_add.setText("")
|
|
||||||
icon3 = QtGui.QIcon()
|
icon3 = QtGui.QIcon()
|
||||||
icon3.addPixmap(QtGui.QPixmap(":/action/ico/edit-add.ico"), QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
icon3.addPixmap(QtGui.QPixmap(":/action/ico/folder-open.ico"), QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
||||||
self.btn_add.setIcon(icon3)
|
self.btn_add_dir.setIcon(icon3)
|
||||||
|
self.btn_add_dir.setObjectName("btn_add_dir")
|
||||||
|
self.vl_edit.addWidget(self.btn_add_dir)
|
||||||
|
self.btn_add = QtWidgets.QPushButton(diag_connections)
|
||||||
|
icon4 = QtGui.QIcon()
|
||||||
|
icon4.addPixmap(QtGui.QPixmap(":/action/ico/edit-add.ico"), QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
||||||
|
self.btn_add.setIcon(icon4)
|
||||||
self.btn_add.setObjectName("btn_add")
|
self.btn_add.setObjectName("btn_add")
|
||||||
self.vl_edit.addWidget(self.btn_add)
|
self.vl_edit.addWidget(self.btn_add)
|
||||||
self.gridLayout.addLayout(self.vl_edit, 0, 1, 1, 1)
|
self.gridLayout.addLayout(self.vl_edit, 0, 1, 1, 1)
|
||||||
|
|
||||||
self.retranslateUi(diag_connections)
|
self.retranslateUi(diag_connections)
|
||||||
self.tab_properties.setCurrentIndex(0)
|
self.tab_properties.setCurrentIndex(0)
|
||||||
self.btn_box.accepted.connect(diag_connections.accept) # type: ignore
|
|
||||||
self.btn_box.rejected.connect(diag_connections.reject) # type: ignore
|
self.btn_box.rejected.connect(diag_connections.reject) # type: ignore
|
||||||
|
self.btn_box.accepted.connect(diag_connections.accept) # type: ignore
|
||||||
QtCore.QMetaObject.connectSlotsByName(diag_connections)
|
QtCore.QMetaObject.connectSlotsByName(diag_connections)
|
||||||
|
|
||||||
def retranslateUi(self, diag_connections):
|
def retranslateUi(self, diag_connections):
|
||||||
|
|||||||
@@ -15,8 +15,9 @@ class Ui_diag_sshauth(object):
|
|||||||
def setupUi(self, diag_sshauth):
|
def setupUi(self, diag_sshauth):
|
||||||
diag_sshauth.setObjectName("diag_sshauth")
|
diag_sshauth.setObjectName("diag_sshauth")
|
||||||
diag_sshauth.setWindowModality(QtCore.Qt.ApplicationModal)
|
diag_sshauth.setWindowModality(QtCore.Qt.ApplicationModal)
|
||||||
diag_sshauth.resize(363, 163)
|
diag_sshauth.resize(331, 225)
|
||||||
self.verticalLayout = QtWidgets.QVBoxLayout(diag_sshauth)
|
self.verticalLayout = QtWidgets.QVBoxLayout(diag_sshauth)
|
||||||
|
self.verticalLayout.setSizeConstraint(QtWidgets.QLayout.SetFixedSize)
|
||||||
self.verticalLayout.setObjectName("verticalLayout")
|
self.verticalLayout.setObjectName("verticalLayout")
|
||||||
self.wid_password = QtWidgets.QWidget(diag_sshauth)
|
self.wid_password = QtWidgets.QWidget(diag_sshauth)
|
||||||
self.wid_password.setObjectName("wid_password")
|
self.wid_password.setObjectName("wid_password")
|
||||||
@@ -44,6 +45,16 @@ class Ui_diag_sshauth(object):
|
|||||||
self.btn_box.setStandardButtons(QtWidgets.QDialogButtonBox.Cancel|QtWidgets.QDialogButtonBox.Ok)
|
self.btn_box.setStandardButtons(QtWidgets.QDialogButtonBox.Cancel|QtWidgets.QDialogButtonBox.Ok)
|
||||||
self.btn_box.setObjectName("btn_box")
|
self.btn_box.setObjectName("btn_box")
|
||||||
self.verticalLayout.addWidget(self.btn_box)
|
self.verticalLayout.addWidget(self.btn_box)
|
||||||
|
self.lbl_info = QtWidgets.QLabel(diag_sshauth)
|
||||||
|
sizePolicy = QtWidgets.QSizePolicy(QtWidgets.QSizePolicy.Expanding, QtWidgets.QSizePolicy.Expanding)
|
||||||
|
sizePolicy.setHorizontalStretch(0)
|
||||||
|
sizePolicy.setVerticalStretch(0)
|
||||||
|
sizePolicy.setHeightForWidth(self.lbl_info.sizePolicy().hasHeightForWidth())
|
||||||
|
self.lbl_info.setSizePolicy(sizePolicy)
|
||||||
|
self.lbl_info.setWordWrap(True)
|
||||||
|
self.lbl_info.setObjectName("lbl_info")
|
||||||
|
self.verticalLayout.addWidget(self.lbl_info)
|
||||||
|
self.verticalLayout.setStretch(3, 1)
|
||||||
|
|
||||||
self.retranslateUi(diag_sshauth)
|
self.retranslateUi(diag_sshauth)
|
||||||
self.btn_box.accepted.connect(diag_sshauth.accept) # type: ignore
|
self.btn_box.accepted.connect(diag_sshauth.accept) # type: ignore
|
||||||
@@ -57,6 +68,7 @@ class Ui_diag_sshauth(object):
|
|||||||
self.lbl_password.setText(_translate("diag_sshauth", "SSH password:"))
|
self.lbl_password.setText(_translate("diag_sshauth", "SSH password:"))
|
||||||
self.cbx_save_password.setToolTip(_translate("diag_sshauth", "Username and password will be saved in secured operating systems\'s password storage."))
|
self.cbx_save_password.setToolTip(_translate("diag_sshauth", "Username and password will be saved in secured operating systems\'s password storage."))
|
||||||
self.cbx_save_password.setText(_translate("diag_sshauth", "Save username and password"))
|
self.cbx_save_password.setText(_translate("diag_sshauth", "Save username and password"))
|
||||||
|
self.lbl_info.setText(_translate("diag_sshauth", "Note: The default user for SSH is \"pi\" which differs from the web configuration. You can find the password on the sticker on the device."))
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|||||||
@@ -6,13 +6,10 @@
|
|||||||
<rect>
|
<rect>
|
||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>418</width>
|
<width>424</width>
|
||||||
<height>97</height>
|
<height>104</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
|
||||||
<string>File transfer...</string>
|
|
||||||
</property>
|
|
||||||
<property name="modal">
|
<property name="modal">
|
||||||
<bool>true</bool>
|
<bool>true</bool>
|
||||||
</property>
|
</property>
|
||||||
|
|||||||
BIN
ui_dev/ico/edit-delete-6.ico
Normal file
BIN
ui_dev/ico/edit-delete-6.ico
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 4.2 KiB |
BIN
ui_dev/ico/process-stop.ico
Executable file → Normal file
BIN
ui_dev/ico/process-stop.ico
Executable file → Normal file
Binary file not shown.
|
Before Width: | Height: | Size: 4.2 KiB After Width: | Height: | Size: 4.2 KiB |
@@ -9,6 +9,7 @@
|
|||||||
<file>ico/file-python.ico</file>
|
<file>ico/file-python.ico</file>
|
||||||
</qresource>
|
</qresource>
|
||||||
<qresource prefix="action">
|
<qresource prefix="action">
|
||||||
|
<file>ico/edit-delete-6.ico</file>
|
||||||
<file>ico/applications-utilities.ico</file>
|
<file>ico/applications-utilities.ico</file>
|
||||||
<file>ico/edit-find.ico</file>
|
<file>ico/edit-find.ico</file>
|
||||||
<file>ico/process-stop.ico</file>
|
<file>ico/process-stop.ico</file>
|
||||||
|
|||||||
@@ -63,16 +63,6 @@
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="6" column="0">
|
|
||||||
<widget class="QLabel" name="lbl_link">
|
|
||||||
<property name="text">
|
|
||||||
<string><html><head/><body><p><a href="https://revpimodio.org/"><span style=" text-decoration: underline; color:#0000ff;">https://revpimodio.org/</span></a></p></body></html></string>
|
|
||||||
</property>
|
|
||||||
<property name="openExternalLinks">
|
|
||||||
<bool>true</bool>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item row="7" column="0" colspan="3">
|
<item row="7" column="0" colspan="3">
|
||||||
<widget class="QDialogButtonBox" name="btn_box">
|
<widget class="QDialogButtonBox" name="btn_box">
|
||||||
<property name="orientation">
|
<property name="orientation">
|
||||||
@@ -127,7 +117,7 @@
|
|||||||
All of them comes with ABSOLUTELY NO WARRANTY, to the extent permitted by
|
All of them comes with ABSOLUTELY NO WARRANTY, to the extent permitted by
|
||||||
applicable law.
|
applicable law.
|
||||||
|
|
||||||
(c) Sven Sager, License: GPLv3</string>
|
(c) Sven Sager, License: GPLv2</string>
|
||||||
</property>
|
</property>
|
||||||
<property name="wordWrap">
|
<property name="wordWrap">
|
||||||
<bool>true</bool>
|
<bool>true</bool>
|
||||||
@@ -147,6 +137,16 @@ applicable law.
|
|||||||
</property>
|
</property>
|
||||||
</spacer>
|
</spacer>
|
||||||
</item>
|
</item>
|
||||||
|
<item row="6" column="0" colspan="2">
|
||||||
|
<widget class="QLabel" name="lbl_link">
|
||||||
|
<property name="text">
|
||||||
|
<string notr="true"><html><head/><body><p><a href="https://revpimodio.org/"><span style=" text-decoration: underline; color:#0000ff;">https://revpimodio.org/</span></a></p></body></html></string>
|
||||||
|
</property>
|
||||||
|
<property name="openExternalLinks">
|
||||||
|
<bool>true</bool>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</widget>
|
</widget>
|
||||||
<resources/>
|
<resources/>
|
||||||
|
|||||||
@@ -149,7 +149,7 @@
|
|||||||
</property>
|
</property>
|
||||||
<layout class="QGridLayout" name="gridLayout_2">
|
<layout class="QGridLayout" name="gridLayout_2">
|
||||||
<item row="0" column="1">
|
<item row="0" column="1">
|
||||||
<widget class="QPushButton" name="btn_aclplcslave">
|
<widget class="QPushButton" name="btn_aclplcserver">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Edit ACL</string>
|
<string>Edit ACL</string>
|
||||||
</property>
|
</property>
|
||||||
@@ -163,14 +163,14 @@
|
|||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="0" column="0">
|
<item row="0" column="0">
|
||||||
<widget class="QCheckBox" name="cbx_plcslave">
|
<widget class="QCheckBox" name="cbx_plcserver">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Start RevPi piControl server</string>
|
<string>Start RevPi piControl server</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="1" column="1">
|
<item row="1" column="1">
|
||||||
<widget class="QLabel" name="lbl_slave_status">
|
<widget class="QLabel" name="lbl_server_status">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>status</string>
|
<string>status</string>
|
||||||
</property>
|
</property>
|
||||||
@@ -180,7 +180,7 @@
|
|||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="1" column="0">
|
<item row="1" column="0">
|
||||||
<widget class="QLabel" name="lbl_lbl_slave_status">
|
<widget class="QLabel" name="lbl_lbl_server_status">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>piControl server is:</string>
|
<string>piControl server is:</string>
|
||||||
</property>
|
</property>
|
||||||
|
|||||||
@@ -108,6 +108,9 @@
|
|||||||
<property name="editable">
|
<property name="editable">
|
||||||
<bool>true</bool>
|
<bool>true</bool>
|
||||||
</property>
|
</property>
|
||||||
|
<property name="sizeAdjustPolicy">
|
||||||
|
<enum>QComboBox::AdjustToContents</enum>
|
||||||
|
</property>
|
||||||
<item>
|
<item>
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string/>
|
<string/>
|
||||||
@@ -218,9 +221,6 @@
|
|||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QPushButton" name="btn_up">
|
<widget class="QPushButton" name="btn_up">
|
||||||
<property name="text">
|
|
||||||
<string/>
|
|
||||||
</property>
|
|
||||||
<property name="icon">
|
<property name="icon">
|
||||||
<iconset resource="ressources.qrc">
|
<iconset resource="ressources.qrc">
|
||||||
<normaloff>:/action/ico/arrow-up.ico</normaloff>:/action/ico/arrow-up.ico</iconset>
|
<normaloff>:/action/ico/arrow-up.ico</normaloff>:/action/ico/arrow-up.ico</iconset>
|
||||||
@@ -229,9 +229,6 @@
|
|||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QPushButton" name="btn_down">
|
<widget class="QPushButton" name="btn_down">
|
||||||
<property name="text">
|
|
||||||
<string/>
|
|
||||||
</property>
|
|
||||||
<property name="icon">
|
<property name="icon">
|
||||||
<iconset resource="ressources.qrc">
|
<iconset resource="ressources.qrc">
|
||||||
<normaloff>:/action/ico/arrow-down.ico</normaloff>:/action/ico/arrow-down.ico</iconset>
|
<normaloff>:/action/ico/arrow-down.ico</normaloff>:/action/ico/arrow-down.ico</iconset>
|
||||||
@@ -240,20 +237,22 @@
|
|||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QPushButton" name="btn_delete">
|
<widget class="QPushButton" name="btn_delete">
|
||||||
<property name="text">
|
|
||||||
<string/>
|
|
||||||
</property>
|
|
||||||
<property name="icon">
|
<property name="icon">
|
||||||
<iconset resource="ressources.qrc">
|
<iconset resource="ressources.qrc">
|
||||||
<normaloff>:/action/ico/edit-delete.ico</normaloff>:/action/ico/edit-delete.ico</iconset>
|
<normaloff>:/action/ico/edit-delete-6.ico</normaloff>:/action/ico/edit-delete-6.ico</iconset>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QPushButton" name="btn_add_dir">
|
||||||
|
<property name="icon">
|
||||||
|
<iconset resource="ressources.qrc">
|
||||||
|
<normaloff>:/action/ico/folder-open.ico</normaloff>:/action/ico/folder-open.ico</iconset>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QPushButton" name="btn_add">
|
<widget class="QPushButton" name="btn_add">
|
||||||
<property name="text">
|
|
||||||
<string/>
|
|
||||||
</property>
|
|
||||||
<property name="icon">
|
<property name="icon">
|
||||||
<iconset resource="ressources.qrc">
|
<iconset resource="ressources.qrc">
|
||||||
<normaloff>:/action/ico/edit-add.ico</normaloff>:/action/ico/edit-add.ico</iconset>
|
<normaloff>:/action/ico/edit-add.ico</normaloff>:/action/ico/edit-add.ico</iconset>
|
||||||
@@ -268,22 +267,6 @@
|
|||||||
<include location="ressources.qrc"/>
|
<include location="ressources.qrc"/>
|
||||||
</resources>
|
</resources>
|
||||||
<connections>
|
<connections>
|
||||||
<connection>
|
|
||||||
<sender>btn_box</sender>
|
|
||||||
<signal>accepted()</signal>
|
|
||||||
<receiver>diag_connections</receiver>
|
|
||||||
<slot>accept()</slot>
|
|
||||||
<hints>
|
|
||||||
<hint type="sourcelabel">
|
|
||||||
<x>248</x>
|
|
||||||
<y>254</y>
|
|
||||||
</hint>
|
|
||||||
<hint type="destinationlabel">
|
|
||||||
<x>157</x>
|
|
||||||
<y>274</y>
|
|
||||||
</hint>
|
|
||||||
</hints>
|
|
||||||
</connection>
|
|
||||||
<connection>
|
<connection>
|
||||||
<sender>btn_box</sender>
|
<sender>btn_box</sender>
|
||||||
<signal>rejected()</signal>
|
<signal>rejected()</signal>
|
||||||
@@ -291,8 +274,8 @@
|
|||||||
<slot>reject()</slot>
|
<slot>reject()</slot>
|
||||||
<hints>
|
<hints>
|
||||||
<hint type="sourcelabel">
|
<hint type="sourcelabel">
|
||||||
<x>316</x>
|
<x>322</x>
|
||||||
<y>260</y>
|
<y>564</y>
|
||||||
</hint>
|
</hint>
|
||||||
<hint type="destinationlabel">
|
<hint type="destinationlabel">
|
||||||
<x>286</x>
|
<x>286</x>
|
||||||
@@ -300,5 +283,21 @@
|
|||||||
</hint>
|
</hint>
|
||||||
</hints>
|
</hints>
|
||||||
</connection>
|
</connection>
|
||||||
|
<connection>
|
||||||
|
<sender>btn_box</sender>
|
||||||
|
<signal>accepted()</signal>
|
||||||
|
<receiver>diag_connections</receiver>
|
||||||
|
<slot>accept()</slot>
|
||||||
|
<hints>
|
||||||
|
<hint type="sourcelabel">
|
||||||
|
<x>254</x>
|
||||||
|
<y>564</y>
|
||||||
|
</hint>
|
||||||
|
<hint type="destinationlabel">
|
||||||
|
<x>157</x>
|
||||||
|
<y>274</y>
|
||||||
|
</hint>
|
||||||
|
</hints>
|
||||||
|
</connection>
|
||||||
</connections>
|
</connections>
|
||||||
</ui>
|
</ui>
|
||||||
|
|||||||
@@ -9,14 +9,17 @@
|
|||||||
<rect>
|
<rect>
|
||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>363</width>
|
<width>331</width>
|
||||||
<height>163</height>
|
<height>225</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
<property name="windowTitle">
|
||||||
<string>SSH authentication</string>
|
<string>SSH authentication</string>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout">
|
<layout class="QVBoxLayout" name="verticalLayout" stretch="0,0,0,1">
|
||||||
|
<property name="sizeConstraint">
|
||||||
|
<enum>QLayout::SetFixedSize</enum>
|
||||||
|
</property>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QWidget" name="wid_password" native="true">
|
<widget class="QWidget" name="wid_password" native="true">
|
||||||
<layout class="QFormLayout" name="formLayout">
|
<layout class="QFormLayout" name="formLayout">
|
||||||
@@ -67,6 +70,22 @@
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QLabel" name="lbl_info">
|
||||||
|
<property name="sizePolicy">
|
||||||
|
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
|
||||||
|
<horstretch>0</horstretch>
|
||||||
|
<verstretch>0</verstretch>
|
||||||
|
</sizepolicy>
|
||||||
|
</property>
|
||||||
|
<property name="text">
|
||||||
|
<string>Note: The default user for SSH is "pi" which differs from the web configuration. You can find the password on the sticker on the device.</string>
|
||||||
|
</property>
|
||||||
|
<property name="wordWrap">
|
||||||
|
<bool>true</bool>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</widget>
|
</widget>
|
||||||
<resources/>
|
<resources/>
|
||||||
|
|||||||
Reference in New Issue
Block a user